一. 哈希對象概述 Redis hash對象是一個string類型的field和value的映射表,hash特別適合用於存儲對象。作為哈希對象的編碼,有二種一是ziplist編碼, 二是hashtable編碼。在不同情況下編碼是可以轉換的。在Redis 中每個 hash 可以存儲 232 - 1 鍵 ...
一. 哈希對象概述
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 。 如果欄位已經存在,舊值被新值覆蓋返回 0 。 127.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"