开发者社区> J3> 正文
阿里云
为了无法计算的价值
打开APP
阿里云APP内打开

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

简介: Redis之Hash超详细API使用及应用场景介绍,不看亏了!
+关注继续查看


一、哈希(Hash)


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

贴张图形象一点


image.png


同样是存储字符串,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存储


微信图片_20220424165348.png


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


  • 以客户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张。


image.png


解决方案

  • 以商家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...


image.png


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


结束语

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


image.png

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
一文搞定Redis五大数据类型及应用场景(2)
一文搞定Redis五大数据类型及应用场景
25 0
【Redis】使用场景
【Redis】使用场景
18 0
js获取浏览器版本
js获取浏览器版本
31 0
Redis 16 大应用场景,竟然这么多。。
例如:热点数据缓存(例如报表、明星出轨),对象缓存、全页缓存、可以提升热点数据的访问数据。
47 0
阿里云智能推荐应用业务场景介绍
我们一定知道手淘的千人千面,对不同的用户针对用户的习惯,喜好给用户推荐展示不同的内容,在本篇文章中,将向大家介绍什么是智能推荐,推荐有哪些行业应用场景,以及基于阿里云智能推荐的产品能力哪些数据需要埋点已经接入智能推荐的流程,从而最终做到“千人千面”。
623 0
OSS Select使用场景及技巧
背景介绍 OSS Select是OSS近期推出的一项新功能,它可以对OSS上的类CSV文件(其他类型文件比如Json也会很快推出)在服务器端运行SQL查询,仅将查询结果返回给客户端。举例来说,假如你有一个10GB的csv日志文件,有一列数据是错误码,想把其中所有错误码等于500的日志找出来,用OSS Select仅仅返回错误码是500的日志,在从而降低用户的数据传输成本以及处理数据的成本,相比在客户端下载整个文件再处理在性能上也可以提高最多到6倍以上。
15764 0
第97天:CSS3渐变和过渡详解
一、渐变 渐变是CSS3当中比较丰富多彩的一个特性,通过渐变我们可以实现许多炫丽的效果,有效的减少图片的使用数量,并且具有很强的适应性和可扩展性。 可分为线性渐变、径向渐变 1、 线性渐变 (gradient 变化)   linear-gradient线性渐变指沿着某条直线朝一个方向产生渐变效果。
995 0
Redis应用场景
Redis开创了一种新的数据存储思路,使用Redis,我们不用在面对功能单调的数据库时,把精力放在如何把大象放进冰箱这样的问题上,而是利用Redis灵活多变的数据结构和数据操作,为不同的大象构建不同的冰箱。
741 0
+关注
J3
不必遗憾。若是美好,叫做精彩。若是糟糕,叫做经历。
67
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
低代码开发师(初级)实战教程
立即下载
阿里巴巴DevOps 最佳实践手册
立即下载
冬季实战营第三期:MySQL数据库进阶实战
立即下载