0
点赞
收藏
分享

微信扫一扫

Redis之Hash超详细API使用及应用场景介绍,不看亏了!




  • j3_liuliang
  • Redis常用API即应用场景系列(Hash),如果觉得有用可以关注博主,不定时更新哦!



相关文章导航​


  1. ​​超详细Redis之Key操作API,什么?看不懂!你来锤我​​
  2. ​​Redis之String超详细API使用及应用场景介绍​​
  3. ​​SCAN及相关SSCAN,HSCAN和ZSCAN命令解析​​
  4. ​​什么?Redis的List类型不会用,看我这个超详细API使用及应用场景​​
  5. ​​Redis之Hash超详细API使用及应用场景介绍,不看亏了!​​
  6. ​​Redis之Set集合数据类型API使用及图文并茂应用场景,不看血亏!​​
  7. ​​Redis之Sorted Set数据类型API及应用场景解析​​

一、哈希(Hash)

实际上主要是对一个对象的多重属性(如人的姓名,性别,年龄)的存储;

贴张图形象一点

Redis之Hash超详细API使用及应用场景介绍,不看亏了!_字段

同样是存储字符串,Hash 与String 的主要区别?

1、把所有相关的值聚集到一个key 中,节省内存空间

2、只使用一个key,减少key 冲突

3、当需要批量获取值的时候,只需要使用一个命令,减少内存/IO/CPU 的消耗

1.1 HSET(hset)

将哈希表 key 中的字段 field 的值设为 value 。


Redis Hset 命令用于为哈希表中的字段赋值 。

如果哈希表不存在,一个新的哈希表被创建并进行 HSET 操作。

如果字段已经存在于哈希表中,旧值将被覆盖。


​语法​

127.0.0.1:6379> HSET KEY_NAME FIELD VALUE

​可以版本:​​>= 2.0.0

​返回值:​​如果字段是哈希表中的一个新建字段,并且值设置成功,返回 1 。 如果哈希表中域字段已经存在且旧值已被新值覆盖,返回 0 。

​案例​

127.0.0.1:6379> flushall
OK
127.0.0.1:6379> hset vipclient name j3_liuliang #设置key为vipclient,域为name,值为j3_liuliang
(integer) 1
127.0.0.1:6379> hget vipclient name #获取key为vipclient,域为name的值
"j3_liuliang"
127.0.0.1:6379> hset website google "www.g.cn" #设置hash类型值
(integer) 1
127.0.0.1:6379> hset website google "www.google.com" #覆盖旧值
(integer) 0
127.0.0.1:6379> hget website google #获取值
"www.google.com"
127.0.0.1:6379>

1.2 HGET(hget)

获取存储在哈希表中指定字段的值


Redis Hget 命令用于返回哈希表中指定字段的值。


​语法​

127.0.0.1:6379> HGET KEY_NAME FIELD_NAME

​可以版本:​​>= 2.0.0

​返回值:​​返回给定字段的值。如果给定的字段或 key 不存在时,返回 nil 。

​案例​

127.0.0.1:6379> flushall
OK
127.0.0.1:6379> hset vipclient name j3_liuliang
(integer) 1
127.0.0.1:6379> hget vipclient name #获取key为vipclient,域为name的值
"j3_liuliang"
127.0.0.1:6379> hget china city #获取不存在的key中的域
(nil)
127.0.0.1:6379>

1.3 HMSET(hmset)

同时将多个 field-value (域-值)对设置到哈希表 key 中。


Redis Hmset 命令用于同时将多个 field-value (字段-值)对设置到哈希表中。

此命令会覆盖哈希表中已存在的字段。

如果哈希表不存在,会创建一个空哈希表,并执行 HMSET 操作。


​语法​

127.0.0.1:6379> HMSET KEY_NAME FIELD1 VALUE1 ...FIELDN VALUEN

​可以版本:​​>= 2.0.0

​返回值:​​如果命令执行成功,返回 OK 。

​案例​

127.0.0.1:6379> flushall
OK
127.0.0.1:6379> hmset vipclient name j3_liuliang age 18 sex 1 #针对一个key的多个域设置值
OK
127.0.0.1:6379> hget vipclient name #获取hash值
"j3_liuliang"
127.0.0.1:6379> hget vipclient age
"18"
127.0.0.1:6379> hget vipclient sex
"1"
127.0.0.1:6379> hmset vipclient age 28 phone 12345678912 #对存在的hash域设置值,结果会覆盖
OK
127.0.0.1:6379> hget vipclient age #覆盖后的值
"28"
127.0.0.1:6379>

1.4 HMGET(hmget)

获取所有给定字段的值


Redis Hmget 命令用于返回哈希表中,一个或多个给定字段的值。

如果指定的字段不存在于哈希表,那么返回一个 nil 值。


​语法​

127.0.0.1:6379> HMGET KEY_NAME FIELD1...FIELDN

​可以版本:​​>= 2.0.0

​返回值:​​一个包含多个给定字段关联值的表,表值的排列顺序和指定字段的请求顺序一样。

​案例​

127.0.0.1:6379> flushall
OK
127.0.0.1:6379> hmset vipclient name j3_liuliang age 18 sex 1 #一次设置多个域值
OK
127.0.0.1:6379> hget vipclient name #获取域值
"j3_liuliang"
127.0.0.1:6379> hmget vipclient name age sex phone id #一次获取多个域值,对没有的域值返回nil
1) "j3_liuliang"
2) "28"
3) "1"
4) "12345678912"
5) (nil)
127.0.0.1:6379>

1.5 HSETNX(hsetnx)

只有在字段 field 不存在时,设置哈希表字段的值。


Redis Hsetnx 命令用于为哈希表中不存在的的字段赋值 。

如果哈希表不存在,一个新的哈希表被创建并进行 HSET 操作。

如果字段已经存在于哈希表中,操作无效。

如果 key 不存在,一个新哈希表被创建并执行 HSETNX 命令。


​语法​

127.0.0.1:6379> HSETNX KEY_NAME FIELD VALUE

​可以版本:​​>= 2.0.0

​返回值:​​设置成功,返回 1 。 如果给定字段已经存在且没有操作被执行,返回 0 。

​案例​

127.0.0.1:6379> flushall
OK
127.0.0.1:6379> hset vipclient name j3_liuliang #创建hash类型key,并设置域和值
(integer) 1
127.0.0.1:6379> hsetnx vipclient age 18 #域不存在,设置成功
(integer) 1
127.0.0.1:6379> hsetnx vipclient age 28 #域存在,不做任何变化
(integer) 0
127.0.0.1:6379> hget vipclient age #域没有变化
"18"
127.0.0.1:6379>

1.6 HGETTALL(hgettall)

获取在哈希表中指定 key 的所有字段和值


Redis Hgetall 命令用于返回哈希表中,所有的字段和值。

在返回值里,紧跟每个字段名(field name)之后是字段的值(value),所以返回值的长度是哈希表大小的两倍。


​语法​

127.0.0.1:6379> HGETALL KEY_NAME

​可以版本:​​>= 2.0.0

​返回值:​​以列表形式返回哈希表的字段及字段值。 若 key 不存在,返回空列表。

​案例​

127.0.0.1:6379> flushall
OK
127.0.0.1:6379> hset vipclient name j3_liuliang #设置key
(integer) 1
127.0.0.1:6379> hsetnx vipclient age 18 #给key添加域并赋值
(integer) 1
127.0.0.1:6379> hgetall vipclient #获取key中的所有域和对应值
1) "name"
2) "j3_liuliang"
3) "age"
4) "18"
127.0.0.1:6379> exists chiji #判断key是否存在
(integer) 0
127.0.0.1:6379> hgetall chiji #获取不存在的key进行获值
(empty list or set)
127.0.0.1:6379>

1.7 HEXISTS(hexists)

查看哈希表 key 中,指定的字段是否存在。


Redis Hexists 命令用于查看哈希表的指定字段是否存在。


​语法​

127.0.0.1:6379> HEXISTS KEY_NAME FIELD_NAME

​可以版本:​​>= 2.0.0

​返回值:​​如果哈希表含有给定字段,返回 1 。 如果哈希表不含有给定字段,或 key 不存在,返回 0 。

​案例​

127.0.0.1:6379> flushall
OK
127.0.0.1:6379> hset vipclient name je_liuliang #给hash的域设置值
(integer) 1
127.0.0.1:6379> hexists vipclient name #判断hash的域是否存在
(integer) 1 #存在
127.0.0.1:6379> hexists vipclient age #判断hash的域是否存在
(integer) 0 #不存在
127.0.0.1:6379>

1.8 HINCRBY(hincrby)

为哈希表 key 中的指定字段的整数值加上增量 increment 。


Redis Hincrby 命令用于为哈希表中的字段值加上指定增量值。

增量也可以为负数,相当于对指定字段进行减法操作。

如果哈希表的 key 不存在,一个新的哈希表被创建并执行 HINCRBY 命令。

如果指定的字段不存在,那么在执行命令前,字段的值被初始化为 0 。

对一个储存字符串值的字段执行 HINCRBY 命令将造成一个错误。

本操作的值被限制在 64 位(bit)有符号数字表示之内。


​语法​

127.0.0.1:6379> HINCRBY KEY_NAME FIELD_NAME INCR_BY_NUMBER

​可以版本:​​>= 2.0.0

​返回值:​​执行 HINCRBY 命令之后,哈希表中字段的值。

​案例​

127.0.0.1:6379> hset vipclient age 18   #设置一个数值hash域age
(integer) 1
127.0.0.1:6379> hincrby vipclient age 10 #给数值域加10,不是数值域会出错
(integer) 28
127.0.0.1:6379> hexists vipclient sex #判断hash域是否存在
(integer) 0
127.0.0.1:6379> hincrby vipclient sex 1 #给不存在的数值域进行加一,先创建域在进行加操作
(integer) 1
127.0.0.1:6379> hget vipclient sex #获取hash域的值
"1"
127.0.0.1:6379>

1.9 HLEN(hlen)

获取哈希表中字段的数量


Redis Hlen 命令用于获取哈希表中字段的数量。


​语法​

127.0.0.1:6379> HLEN KEY_NAME

​可以版本:​​>= 2.0.0

​返回值:​​哈希表中字段的数量。 当 key 不存在时,返回 0 。

​案例​

127.0.0.1:6379> exists vipclient  #判断key是否存在
(integer) 1
127.0.0.1:6379> hlen vipclient #获取存在的hash的域数量
(integer) 3
127.0.0.1:6379> exists ordy #判断key是否存在
(integer) 0
127.0.0.1:6379> hlen ordy #获取不存在的hash的域数量
(integer) 0
127.0.0.1:6379>

1.10 HDEL(hdel)

删除一个或多个哈希表字段


Redis Hdel 命令用于删除哈希表 key 中的一个或多个指定字段,不存在的字段将被忽略。


​语法​

127.0.0.1:6379> HDEL KEY_NAME FIELD1.. FIELDN

​可以版本:​​>= 2.0.0

​返回值:​​被成功删除字段的数量,不包括被忽略的字段。

​案例​

127.0.0.1:6379> hlen vipclient          #获取hash的域数量
(integer) 3
127.0.0.1:6379> hdel vipclient age #删除指定hash的域
(integer) 1
127.0.0.1:6379> hlen vipclient #再次获取hash的域数量
(integer) 2
127.0.0.1:6379> hdel vipclient name sex age #一次删除多个hash的域,对不存在的域,进行忽略
(integer) 2
127.0.0.1:6379> hlen vipclient #再次获取hash的域数量
(integer) 0
127.0.0.1:6379>

1.11 HVALS(hvals)

获取哈希表中所有值


Redis Hvals 命令返回哈希表所有字段的值。


​语法​

127.0.0.1:6379> HVALS KEY_NAME FIELD VALUE

​可以版本:​​>= 2.0.0

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

​案例​

127.0.0.1:6379> flushall
OK
127.0.0.1:6379> hmset vipclient name j3_liulinag age 18 #一次设置多个域值
OK
127.0.0.1:6379> hvals vipclient #获取所有域值列表
1) "j3_liulinag"
2) "18"
127.0.0.1:6379> exists ordy #判断key是否存在
(integer) 0
127.0.0.1:6379> hvals ordy #获取不存在的key域值列表
(empty list or set) #空
127.0.0.1:6379>

1.12 HINCRBYFLOAT(hincrbyfloat)

为哈希表 key 中的指定字段的浮点数值加上增量 increment 。


Redis Hincrbyfloat 命令用于为哈希表中的字段值加上指定浮点数增量值。

如果指定的字段不存在,那么在执行命令前,字段的值被初始化为 0 。


​语法​

127.0.0.1:6379> HINCRBYFLOAT KEY_NAME FIELD_NAME INCR_BY_NUMBER

​可以版本:​​>= 2.6.0

​返回值:​​执行 Hincrbyfloat 命令之后,哈希表中字段的值。

​案例​

127.0.0.1:6379> hset vipclient money 99.5       #创建money域值,值为99.5
(integer) 1
127.0.0.1:6379> hincrbyfloat vipclient money 0.3 #给money 加 0.3
"99.8"
127.0.0.1:6379> hincrbyfloat vipclient money -0.4 #给money 加 -0.4
"99.4"
127.0.0.1:6379>

1.13 HKEYS(hkeys)

获取所有哈希表中的字段


Redis Hkeys 命令用于获取哈希表中的所有字段名。


​语法​

127.0.0.1:6379> HKEYS KEY_NAME FIELD_NAME INCR_BY_NUMBER

​可以版本:​​>= 2.0.0

​返回值:​​包含哈希表中所有字段的列表。 当 key 不存在时,返回一个空列表。

​案例​

127.0.0.1:6379> flushall
OK
127.0.0.1:6379> hmset vipclient name j3_liuliang age 18 #设置多个域并赋值
OK
127.0.0.1:6379> hkeys vipclient #获取hash的所有域名称
1) "name"
2) "age"
127.0.0.1:6379> exists ordy #判断key是否存在
(integer) 0
127.0.0.1:6379> hkeys ordy #获取不存在的hash的域名称
(empty list or set) #空
127.0.0.1:6379>

二、应用场景

2.1 购物车

业务分析


  • 分析购物车的redis存储模型

添加、浏览、更改数量、删除、清空



购物车和数据库之间持久化同步(暂不考虑)
购物车与订单间关系(暂不考虑)

提交购物车:读取数据生成订单

商家临时价格调整:隶属于订单级别



未登陆用户购物车信息存储(暂不考虑)

cookie存储




Redis之Hash超详细API使用及应用场景介绍,不看亏了!_应用场景_02

从上图我们可以分析得出最终解决方案


  • 以客户id作为key,每位客户创建一个hash存储结构存储对应的购物车信息
  • 将商品编号作为field,购买数量作为value进行存储
  • 添加商品:追加全新的field与value
  • 浏览:遍历hash
  • 更改数量:自增/自减,设置value值
  • 删除商品:删除field
  • 清空:删除key
  • 全选:hgetall
  • 购物车总数量:hlen
  • 增加某件商品的数量:hincrby

思考:当前设计是否加速了购物车的呈现

当前仅仅是将数据存储到redis中,并没有起到加速的作用,商品信息还要二次查询数据库


  • 每条购物车中的商品记录保存成两条field
  • field1专用于保存购买数量

命名格式:商品id:nums

保存数据:数值



field2专用于保存购物车中显示的信息,包含文字描述,图片地址,所属商家信息等

命名格式:商品id:info

保存数据:json




​比如:hmset 用户id 商品1:nums 数量5个 商品1:info {json字符串}​

有一个问题就是:

多个用户购买商品1,商品1信息都会存储一次,商品1的info​​重复​​​,所以​​将field2作为一个独立hash,公共hash​​。但是还会有一个问题就是每个用户都会在公共hash表中添加商品信息,我们用如下命令解决这个问题。这个命令是有重复的就不添加,如果商品信息改变我们直接用hmset更新覆盖原信息。这样我们用hash存储的就只是数量或者商品编号了。

hsetnx key field value  #存在就不做改变

1.2 节日抢购

hash还可以用于抢购、限购、限量发放优惠券、激活码等业务

业务场景

双十一活动日,销售手机充值卡的商家对移动、联通、电信的30元、50元、100元商品推出抢购活动,每种商品抢购上限为1000张。

Redis之Hash超详细API使用及应用场景介绍,不看亏了!_redis_03

解决方案


  • 以商家id作为key
  • 将参与抢购的商品id作为field
  • 将参与抢购的商品数量作为对应的value
  • 抢购时使用降值得方式控制产品数量
  • 实际业务中还有超卖等实际问题,此处暂不考虑

hmset p01 c30 1000 c50 1000 c100 1000

p01:商家

c30:30元充值卡

1000:1000张

被买走一张c30充值卡

hincrby p01 c30 -1

1.3 对象缓存

业务分析

项目中,实例化的对象都是被对象名所引用这,并且对象中的各个属性和值是一一对应的;所以要将对象缓存下来的话,格式可以如下:

hmset 对象名称 属性1 值1 属性2 值2...

Redis之Hash超详细API使用及应用场景介绍,不看亏了!_其他_04

到此hash的应用场景就介绍到这里了;

结束语


  • 本文结合​​Redis中文网​​和博主的实践案例所写,下期写Set类型
  • 由于博主才疏学浅,难免会有纰漏,假如你发现了错误或偏见的地方,还望留言给我指出来,我会对其加以修正。
  • 如果你觉得文章还不错,你的转发、分享、点赞、留言就是对我最大的鼓励。
  • 感谢您的阅读,十分欢迎并感谢您的关注。

Redis之Hash超详细API使用及应用场景介绍,不看亏了!_其他_05



举报

相关推荐

0 条评论