redis 系列12 哈希對象


一. 哈希對象概述

  Redis hash對象是一個string類型的field和value的映射表,hash特別適合用於存儲對象。作為哈希對象的編碼,有二種一是ziplist編碼, 二是hashtable編碼。在不同情況下編碼是可以轉換的。在Redis 中每個 hash 可以存儲 232 - 1 鍵值對(40多億)。

  1.1 ziplist編碼

    如果哈希對象是以ziplist編碼的壓縮列表作為底層實現,那么每當有新的鍵值對要加入到哈希對象時,程序會先將保存了鍵的壓縮列表節點推入到壓縮列表表尾,然后再保存了值的壓縮列表節點推入到壓縮列表表尾。因此:

    (1)保存了同一個鍵值對的兩個節點總是緊挨在一起,保存鍵的節點在前,保存值的節點在后。

    (2)先添加到哈希對象中的鍵值對被放在壓縮列表的表頭方向,后添加到哈希對象中的鍵值對會被放在壓縮列表的表尾方向。

    例1 下面使用hset命令,創建一個哈希對象名為profile。這個值對象使用ziplist編碼。對象所使用的壓縮列表底層實現 如下圖所示:

  127.0.0.1:6379> hset profile name "tom"
  (integer) 1
  127.0.0.1:6379> hset profile age 25
  (integer) 1
  127.0.0.1:6379> object encoding profile
  "ziplist"

        

  1.2  hashtable編碼

    例2 下面還是使用hset命令,創建一個哈希對象名為book。這個值對象使用hashtable編碼,因為字符串長度大於64字節,對象所使用字典底層實現要比ziplist編碼的壓縮列表底層實現結構更為復雜。這里就在再貼ziplist編碼的redisobject結構,代碼如下所示:

  127.0.0.1:6379> hset 
  book _long_long_long_long_long_long_long_long_long_long_long_long_long_long_long_long_long_long_long_long "content"
  (integer) 1
  127.0.0.1:6379> object encoding book
  "hashtable"

  1.3 編碼轉換

    哈希對象可以同時滿足以下兩個條件時,哈希對象使用ziplist編碼:(1)哈希對象保存的所有鍵值對的鍵和值的字符串長度都小於64字節;(2) 哈希對象保存的鍵值對數量小於512個。 當不能滿足這兩個條件的哈希對象需要使用hashtable編碼。

    對於上面編碼轉換的兩個條件,上限值是可以修改的,具體看配置文件中關於hash-max-ziplist-value選項和hash-max-ziplist-entries選項說明。

  127.0.0.1:6379> config get hash-max-ziplist-value
  1) "hash-max-ziplist-value"
  2) "64"
  127.0.0.1:6379> config get hash-max-ziplist-entries
  1) "hash-max-ziplist-entries"
  2) "512"

 

二. 哈希命令實現

 -- Hdel 命令用於刪除哈希表 key 中的一個或多個指定字段,不存在的字段將被忽略。返回被成功刪除字段的數量,不包括被忽略的字段。
    127.0.0.1:6379> hdel myhash field1 field2
    (integer) 2
    127.0.0.1:6379> hgetall myhash 
    (empty list or set) --已刪除field1 field2

 --Hexists 命令用於查看哈希表的指定字段是否存在。
    127.0.0.1:6379> hexists profile field1
    (integer) 0
    127.0.0.1:6379> hset profile field1 "one"
    (integer) 1  
    127.0.0.1:6379> hexists profile field1
    (integer) 1 --字段field1存在,返回1

 -- Hget 命令用於返回哈希表中指定字段的值
    127.0.0.1:6379> hget profile field1
    "one"

  -- Hgetall 命令用於返回哈希表中,所有的字段和值。
    127.0.0.1:6379> hgetall profile
    1) "field1"
    2) "one"
    3) "field2"
    4) "two"

-- Hincrby 命令用於為哈希表中的字段值加上指定增量值。增量也可以為負數,相當於對指定字段進行減法操作。
    127.0.0.1:6379> hset myhash field1 20
    (integer) 1
    127.0.0.1:6379> hincrby myhash field1 1
    (integer) 21

-- Hincrbyfloat 命令用於為哈希表中的字段值加上指定浮點數增量值。
    127.0.0.1:6379> HSET myhash field 20.50
    (integer) 1
    127.0.0.1:6379> hincrbyfloat myhash field 0.1
    20.6"

-- Hkeys 命令用於獲取哈希表中的所有字段名。
    127.0.0.1:6379> hkeys profile
    1) "field1"
    2) "field2"

--Hlen 命令用於獲取哈希表中字段的數量。
    127.0.0.1:6379> hlen profile
    (integer) 2

 -- Hmget 命令用於返回哈希表中,一個或多個給定字段的值
    127.0.0.1:6379> hmget profile field1 field2 field
    1) "one"
    2) "two"
    3) (nil)  --這個字段不存在

-- Hmset 命令用於同時將多個 field-value (字段-值)對設置到哈希表中
    127.0.0.1:6379>  HmSET myhash field1 "foo" field2 "bar"
    OK
    127.0.0.1:6379> hgetall myhash 
    1) "field1"
    2) "foo"
    3) "field2"
    4) "bar"
 
-- Hset 命令用於為哈希表中的字段賦值 ,如果字段已經存在於哈希表中,舊值將被覆蓋。如果字段是一個新建字段,值設置成功返回 1 。 如果字段已經存在,舊值被新值覆蓋返回 0127.0.0.1:6379> hset myhash field1 "three"
    (integer) 0 -- 已存在的覆蓋值成功,返回0
    127.0.0.1:6379> hgetall myhash
    1) "field1"
    2) "three"
    3) "field2"
    4) "bar"
    
-- Hsetnx 命令用於為哈希表中不存在的的字段賦值,如果字段已經存在於哈希表中,操作無效。 
    127.0.0.1:6379> HSETNX myhash field1 "foo"
    (integer) 1
    127.0.0.1:6379> HSETNX myhash field1 "bar"
    (integer) 0  -- 已存在的字段操作無效

-- Hvals 命令返回哈希表所有字段的值。
    127.0.0.1:6379> hvals myhash
    1) "foo"

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM