2.5 hash(字典)

  Redis的字典结构为数组+链表⼆维结构。第⼀维hash的数组位置碰撞时,就会将碰撞的元素使⽤链表串接起来。Redis的字典的值只能是字符串。当字典很大的时候,会进行rehash,Redis为了⾼性能,不能堵塞服务,采⽤了渐进式rehash策略。

  渐进式rehash保留新旧两个hash结构,查询时会同时查询两个hash结构,然后在后续的定时任务中以及hash操作指令中,循序渐进地将旧hash的内容⼀点点迁移到新的hash结构中。当搬迁完成了,就会使⽤新的hash结构取⽽代之。当hash移除了最后⼀个元素之后,该数据结构⾃动被删除,内存被回收。

使用场景

  下面我们看一下它的API,所有hash的命令都是h开头

1. HSET hash field value

时间复杂度: O(1)

  将哈希表hash中域field的值设置为value。如果给定的哈希表并不存在,那么一个新的哈希表将被创建并执行HSET操作。如果域field已经存在于哈希表中,那么它的旧值将被新值value覆盖。当HSET命令在哈希表中新创建field域并成功为它设置值时,命令返回1;如果域field已经存在于哈希表,并且HSET命令成功使用新值覆盖了它的旧值,那么命令返回 0 。

设置一个新域:

redis> HSET website google "www.g.cn"
(integer) 1

redis> HGET website google
"www.g.cn"

对一个已存在的域进行更新:

redis> HSET website google "www.google.com"
(integer) 0

redis> HGET website google
"www.google.com"

2.HGET hash field

时间复杂度:O(1)

  返回哈希表中给定域的值。HGET命令在默认情况下返回给定域的值。如果给定域不存在于哈希表中,又或者给定的哈希表并不存在,那么命令返回nil。

  • 域存在的情况:
redis> HSET homepage redis redis.com
(integer) 1

redis> HGET homepage redis
"redis.com"
  • 域不存在的情况:
redis> HGET site mysql
(nil)

3.HDEL

HDEL key field [field …]

  O(N),N为要删除的域的数量。 删除哈希表key中的一个或多个指定域,不存在的域将被忽略。返回值为被成功移除的域的数量,不包括被忽略的域。

# 测试数据
redis> HGETALL abbr
1) "a"
2) "apple"
3) "b"
4) "banana"
5) "c"
6) "cat"
7) "d"
8) "dog"


# 删除单个域
redis> HDEL abbr a
(integer) 1


# 删除不存在的域
redis> HDEL abbr not-exists-field
(integer) 0


# 删除多个域
redis> HDEL abbr b c
(integer) 2

redis> HGETALL abbr
1) "d"
2) "dog"

4. HSETNX hash field value

  时间复杂度:O(1)

  当且仅当域field尚未存在于哈希表的情况下,将它的值设置为value。如果给定域已经存在于哈希表当中,那么命令将放弃执行设置操作。如果哈希表hash 不存在,那么一个新的哈希表将被创建并执行HSETNX命令。HSETNX命令在设置成功时返回1,在给定域已经存在而放弃执行设置操作时返回 0 。

  • 域尚未存在,设置成功:
redis> HSETNX database key-value-store Redis
(integer) 1

redis> HGET database key-value-store
"Redis"
  • 域已经存在,设置未成功,域原有的值未被改变:
redis> HSETNX database key-value-store Riak
(integer) 0

redis> HGET database key-value-store
"Redis"

5. HLEN

  时间复杂度:O(1)

  返回哈希表key中域的数量。当key不存在时,返回0。

redis> HSET db redis redis.com
(integer) 1

redis> HSET db mysql mysql.com
(integer) 1

redis> HLEN db
(integer) 2

redis> HSET db mongodb mongodb.org
(integer) 1

redis> HLEN db
(integer) 3

6.HMSET

HMSET key field value [field value …]

  时间复杂度:O(N,N为field-value对的数量。

  同时将多个field-value(域-值)对设置到哈希表key中。此命令会覆盖哈希表中已存在的域。如果key 不存在,一个空哈希表被创建并执行 HMSET 操作。如果命令执行成功,返回 OK 。当 key 不是哈希表(hash)类型时,返回一个错误。

redis> HMSET website google www.google.com yahoo www.yahoo.com
OK

redis> HGET website google
"www.google.com"

redis> HGET website yahoo
"www.yahoo.com"

7. HMGET

HMGET key field [field …]

  时间复杂度:O(N),N为给定域的数量。

  返回哈希表key中,一个或多个给定域的值。

  如果给定的域不存在于哈希表,那么返回一个nil值。因为不存在的key 被当作一个空哈希表来处理,所以对一个不存在的key进行HMGET操作将返回一个只带有nil值的表。具体返回一个包含多个给定域的关联值的表,表值的排列顺序和给定域参数的请求顺序一样。

redis> HMSET pet dog "doudou" cat "nounou"    # 一次设置多个域
OK

redis> HMGET pet dog cat fake_pet             # 返回值的顺序和传入参数的顺序一样
1) "doudou"
2) "nounou"
3) (nil)                                      # 不存在的域返回nil值

8.HINCRBY

HINCRBY key field increment

  时间复杂度:O(1)

  为哈希表key中的域field的值加上增量increment。增量也可以为负数,相当于对给定域进行减法操作。如果key不存在,一个新的哈希表被创建并执行HINCRBY命令。如果域field不存在,那么在执行命令前,域的值被初始化为0。对一个储存字符串值的域field执行HINCRBY命令将造成一个错误。本操作的值被限制在64位(bit)有符号数字表示之内。执行HINCRBY命令之后,返回值哈希表key中域field的值。

# increment 为正数
redis> HEXISTS counter page_view    # 对空域进行设置
(integer) 0

redis> HINCRBY counter page_view 200
(integer) 200

redis> HGET counter page_view
"200"


# increment 为负数
redis> HGET counter page_view
"200"

redis> HINCRBY counter page_view -50
(integer) 150

redis> HGET counter page_view
"150"


# 尝试对字符串值的域执行HINCRBY命令
redis> HSET myhash string hello,world       # 设定一个字符串值
(integer) 1

redis> HGET myhash string
"hello,world"

redis> HINCRBY myhash string 1              # 命令执行失败,错误。
(error) ERR hash value is not an integer

redis> HGET myhash string                   # 原值不变
"hello,world"

9. HKEYS

时间复杂度:O(N),N为哈希表的大小。

  返回哈希表key中的所有域。即一个包含哈希表中所有域的表。当key不存在时,返回一个空表。

# 哈希表非空
redis> HMSET website google www.google.com yahoo www.yahoo.com
OK

redis> HKEYS website
1) "google"
2) "yahoo"

# 空哈希表/key不存在
redis> EXISTS fake_key
(integer) 0

redis> HKEYS fake_key
(empty list or set)

10.HVALS

HVALS key

  时间复杂度:O(N),N为哈希表的大小。返回哈希表key中所有域的值。即一个包含哈希表中所有值的表。当 key 不存在时,返回一个空表。

# 非空哈希表
redis> HMSET website google www.google.com yahoo www.yahoo.com
OK

redis> HVALS website
1) "www.google.com"
2) "www.yahoo.com"


# 空哈希表/不存在的key
redis> EXISTS not_exists
(integer) 0

redis> HVALS not_exists
(empty list or set)

11.HGETALL

HGETALL key

  时间复杂度:O(N),N为哈希表的大小。

  返回哈希表key中,所有的域和值。在返回值里,紧跟每个域名(field name)之后是域的值(value),所以返回值的长度是哈希表大小的两倍。返回值以列表形式返回哈希表的域和域的值。若key不存在,返回空列表。

redis> HSET people jack "Jack Sparrow"
(integer) 1

redis> HSET people gump "Forrest Gump"
(integer) 1

redis> HGETALL people
1) "jack"          # 域
2) "Jack Sparrow"  # 值
3) "gump"
4) "Forrest Gump"

小心单线程,数据量大的话,会比较慢

12. hsetnx

  时间复杂度:O(1)

  当且仅当域field尚未存在于哈希表的情况下,将它的值设置为value。如果给定域已经存在于哈希表当中,那么命令将放弃执行设置操作。如果哈希表hash 不存在,那么一个新的哈希表将被创建并执行HSETNX命令。HSETNX命令在设置成功时返回1,在给定域已经存在而放弃执行设置操作时返回0。

redis> HSETNX database key-value-store Redis
(integer) 1

redis> HGET database key-value-store
"Redis"
  • 域尚未存在,设置成功:
redis> HSETNX database key-value-store Redis
(integer) 1

redis> HGET database key-value-store
"Redis"
  • 域已经存在,设置未成功,域原有的值未被改变:
redis> HSETNX database key-value-store Riak
(integer) 0

redis> HGET database key-value-store
"Redis"

13. hincrbyfloat

HINCRBYFLOAT key field increment

  时间复杂度:O(1)为哈希表 key中的域field加上浮点数增量increment 。如果哈希表中没有域field,那么HINCRBYFLOAT会先将域field的值设为0,然后再执行加法操作。如果键key不存在,那么HINCRBYFLOAT会先创建一个哈希表,再创建域field,最后再执行加法操作。当以下任意一个条件发生时,返回一个错误:

  • 域field的值不是字符串类型(因为redis 中的数字和浮点数都以字符串的形式保存,所以它们都属于字符串类型)
  • 域field当前的值或给定的增量increment不能解释(parse)为双精度浮点数(double precision floating point number)

  返回值为:执行加法操作之后field域的值。

# 值和增量都是普通小数
redis> HSET mykey field 10.50
(integer) 1
redis> HINCRBYFLOAT mykey field 0.1
"10.6"


# 值和增量都是指数符号
redis> HSET mykey field 5.0e3
(integer) 0
redis> HINCRBYFLOAT mykey field 2.0e2
"5200"


# 对不存在的键执行 HINCRBYFLOAT
redis> EXISTS price
(integer) 0
redis> HINCRBYFLOAT price milk 3.5
"3.5"
redis> HGETALL price
1) "milk"
2) "3.5"


# 对不存在的域进行 HINCRBYFLOAT
redis> HGETALL price
1) "milk"
2) "3.5"
redis> HINCRBYFLOAT price coffee 4.5   # 新增 coffee 域
"4.5"
redis> HGETALL price
1) "milk"
2) "3.5"
3) "coffee"
4) "4.5"

  知道上面的命令后,就可以做一些事情了。

应用

  类似于字符串,我们可以记录网站每个用户个人主页的访问量

hincrby user chenxiao pageviewCount

  当然还有缓存用户信息。对于记录个人主页的访问量,自然字符串要比hash更好点。但是对于缓存用户新信息这种逻辑要好好斟酌一下。

  字符串Key:Value的结构:(第一种方案 String-v1)

key: 'user:userId'
value:
{
    "name": "chenxiao",
    "age":100,
  "pageview": 8000000
}

value是序列化的结果

  字符串Key:Value的结构:(第二种方案String-v2)

key: user:userId:name
value: chenxiao

key: user:userId:age
value: 100

key: user:userId:pageView
value: 800000

  相比上面的方案更新属性更方便,只需要一条

  再看看hash形式的方案(hash)

key:user:userId

field:name
value:chenxiao

field:age
value:100

field:pageView
vale:80000

  3种方案比较:

方案 优点 缺点
String v1 编程简单,可能节约内存 1.序列化开销 2.设置属性要操作整个数据。
String v2 直观,可以部分更新 1.内存占用较大 2.key较为分散
hash 直观 节省空间 可以部分更新 1.编程稍微复杂 2.ttl不好控制
版权声明: 本文为智客工坊「沉晓」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

results matching ""

    No results matching ""