1. 介绍
Hash 是一个键值对(key - value)集合,其中 value 形式如: value = [{field1, value1}, ...{fieldN, valueN}]
。Hash 特别适合用于存储对象。
Hash 和 String 对象的区别如下:
Redis String
uid:1:name ------------> tom
uid:1:age ------------> 23
Redis Hash
uid:1 ------------> 丨name: tom丨
丨age: 23 丨
2. 内部实现
Hash 类型的底层数据结构是由压缩列表或哈希表实现的:
-
如果哈希类型元素个数小于
512
个(默认值,可由hash-max-ziplist-entries
配置),所有值小于64
字节(默认值,可由hash-max-ziplist-value
配置)的话,Redis 会使用压缩列表作为 Hash 类型的底层数据结构; -
如果哈希类型元素不满足上面条件,Redis 会使用哈希表作为 Hash 类型的底层数据结构。
在 Redis 7.0 中,压缩列表数据结构已经废弃了,交由 listpack 数据结构来实现了。
3. 常用命令
# 存储一个哈希表 key 的键值
HSET key field value
# 获取哈希表 key 对应的 field 键值
HGET key field
# 在一个哈希表 key 中存储多个键值对
HMSET key field value [field value...]
# 批量获取哈希表 key 中多个 field 键值
HMGET key field [fiedl ...]
# 删除哈希表 key 中的 field 键值
HDEL key field [field ...]
# 返回哈希表 key 中的 field 的数量
HLEN key
# 返回哈希表 key 中所有的键值
HGETALL key
# 为哈希表 key 中 field 键的值加上增量 n
HINCRBY key field n
4. 应用场景
4.1 缓存对象
Hash 类型的(key,field,value)的结构与对象的(对象id,属性,值)的结构相似,也可以用来存储对象。
我们以用户信息为例,它在关系型数据库中的结构是这样的:
uid name age
1 Tom 15
2 Jerry 12
我们可以使用如下命令,将用户对象的信息存储到 Hash 类型:
# 存储一个哈希表 uid:1 的键值
> HMSET uid:1 name Tom age 15
2
# 存储一个哈希表 uid:2 的键值
> HMSET uid:2 name Jerry age 13
2
# 获取哈希表用户id 为 1 中所有的键值
> HGETALL uid:1
1) "name"
2) "Tom"
3) "age"
4) "15"
在介绍 String 类型的应用场景时有所介绍,String + Json 也是存储对象的一种方式,那么存储对象时,到底用 String + Json 还是用 Hash 呢?
一般对象用 String + Json 存储,对象中某些频繁变化的属性可以考虑抽出来用 Hash 类型存储。
4.2 购物车
以用户 id 为 key,商品 id 为 field,商品数量为 value,恰好构成了购物车的 3 个要素:
设计的命令如下:
-
添加商品:
HSET cart:{用户id}{商品id} 1
-
添加数量:
HINCRBY cart:{用户id}{商品id} 1
-
商品总数:
HLEN cart:{用户id}
-
删除商品:
HDEL cart:{用户id}{商品id}
-
获取购物车所有商品:
HGETALL cart:{用户id}
当前仅仅是将商品 Id 存储到了 Redis 中,在回显商品具体信息的时候,还需要拿着商品 id 查询一次数据库,获取完成的商品的信息。