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

本文详细介绍了Redis中的Hash数据类型及其常用API,包括HSET、HGET、HMSET、HMGET、HSETNX、HGETALL、HEXISTS、HINCRBY、HLEN、HDEL、HVALS、HINCRBYFLOAT、HKEYS等,适用于存储对象属性、购物车管理、抢购业务等多种场景。
摘要由CSDN通过智能技术生成
  • 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)

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

贴张图形象一点

在这里插入图片描述

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

在这里插入图片描述

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

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

在这里插入图片描述

解决方案

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

在这里插入图片描述

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

结束语

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

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

J3code

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值