Redis 数据类型 Hash

RedisJSON数据的支持不是很友好。通常把JSON转成String存储到Redis中,但现在的JSON数据都是连环嵌套的,每次更新时都要先获取整个JSON,然后更改其中一个字段再放上去。
这种使用方式,如果在海量的请求下,JSON字符串比较复杂,会导致在频繁更新数据使网络I/O跑满,甚至导致系统超时、崩溃。
所以Redis官方推荐采用Hash(字典)保存对象。但是Hash结构的存储消耗要高于单个字符串。

Redis的字典和JavaHashMap类似,它是无序字典。包括内部结构的实现也和HashMap也是一致的,同样的数组 + 链表二维结构。

存取

HSET

设置字典key值的field字段值为value

1
HSET key field value

如果key不存在,创建key并进行HSET操作。
如果field不存在,则增加新字段,设置成功,返回1。如果field已存在,覆盖旧值,返回0

1
2
3
4
5
6
7
8
redis> hset student name xiaoming
(integer) 1
redis> hget student name
"xiaoming"
redis> hset student age 18
(integer) 0
redis> hget student age
"18"

HSETNX

HSET一样,但是只在字段field不存时才会设置。设置成功,返回1。失败,返回0

1
HSETNX key field value

如果field字段已经存在,该操作无效,返回0

1
2
3
4
5
redis> HSETNX student phone 16790624749
(integer) 1
redis> HSETNX nosql phone 16790624778
(integer) 0

HGET

获取指定字段field的值。

1
HSETNX key field value

key存在且field存在则返回其值,否则返回nil

1
2
3
4
5
redis> HGET student address
(nil)
redis> HGET student phone
"16790624749"

HGETALL

获取key所有字段的值。

1
HSETNX key field value

以列表形式返回哈希表中的字段和值。若哈希表不存在,否则返回一个空列表。

1
2
3
4
5
6
7
8
9
10
11
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"

批量操作

HashString一样,支持操作多个字段。

HMSET

将一个或多个field/value对设置到哈希表key

1
HMSET key field value [field value ...]

如果key不存在,创建key并进行HMSET操作。
如果field不存在,则增加新字段,设置成功,返回1。如果field已存在,覆盖旧值,返回0

1
2
3
4
5
6
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"

HMGET

返回key中一个或多个指定字段的值。

1
HMGET key field [field ...]

返回一个列表,如果指定的字段在不存在,则返回一个nil。如果key不存在将返回一个只带有nil值的表。如果key为非Hash结构,则返回一个错误。

1
2
3
4
5
6
7
8
9
10
11
12
13
redis> HMSET pet dog "doudou" cat "nounou"
OK
redis> HMGET pet dog cat fake_pet
1) "doudou"
2) "nounou"
3) (nil) # 不存在,返回nil值
# 非Hash结构
redis> set istring "Hello"
OK
redis> hmget istring notexsitfield
(error) WRONGTYPE Operation against a key holding the wrong kind of value

自增

HINCRBY

key中的指定字段field增加一个增量increment

1
HINCRBY key field increment

increment可以为负数,相当于对进行减法操作。
如果key不存在,创建key,并执行HINCRBY操作。如果指定field不存在,先初始化为0,再执行HINCRBY。如果为非数字值执行HINCRBY操作,则返回一个错误。
本操作的值被限制在64位(bit)有符号数字表示之内。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
# 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"

HINCRBYFLOAT

HINCRBY一样,不同的是HINCRBYFLOAT是为指定字段field增加一个浮点数增量increment

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
# 值和增量都是普通小数
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
"4.5"
redis> HGETALL price
1) "milk"
2) "3.5"
3) "coffee"
4) "4.5"

其他

HDEL

删除key中的一个或多个字段。

1
HDEL key field [field ...]

不存在的字段将被忽略。返回被成功删除的字段数量。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
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"

HEXISTS

判断key中指定的field是否存在。

1
HEXISTS key field

指定字段存在,返回1。不存在,返回0

1
2
3
4
5
6
redis> HEXISTS phone myphone
(integer) 0
redis> HSET phone myphone nokia-1110
(integer) 1
redis> HEXISTS phone myphone
(integer) 1

HLEN

返回key哈希表的长度,也就是所有字段的数量。

1
HLEN key

哈希表存在,返回字段数。哈希表不存在,返回0

1
2
3
4
5
6
7
8
9
10
11
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

HKEYS

返回key中的所有字段。

1
HKEYS key

哈希表存在,返回字段列表。哈希表不存在,返回空列表。

1
2
3
4
5
6
7
8
9
10
11
12
# 哈希表非空
redis> HMSET website google www.google.com yahoo www.yahoo.com
OK
redis> HKEYS website
1) "google"
2) "yahoo"
# 空哈希表不存在
redis> EXISTS fake_key
(integer) 0
redis> HKEYS fake_key
(empty list or set)

HVALS

HKEYS对应,HVALS返回key中的所有字段的值。

1
HVALS key

哈希表存在,返回字段数。哈希表不存在,返回0

1
2
3
4
5
6
7
8
9
10
11
12
13
# 非空哈希表
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)

HSTLEN

返回key中指定fieldvalue的字符串长度。

1
HSTLEN key field

如果key或者field不存在,返回0

1
2
3
4
5
6
7
8
redis> HMSET myhash f1 HelloWorld f2 99 f3 -256
OK
redis> HSTRLEN myhash f1
(integer) 10
redis> HSTRLEN myhash f2
(integer) 2
redis> HSTRLEN myhash f3
(integer) 4

HSCAN

参考SCAN命令。

1
HSCAN key cursor [MATCH pattern] [COUNT count]