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

本文涉及的产品
云数据库 Redis 版,社区版 2GB
推荐场景:
搭建游戏排行榜
简介: 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实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore     ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库 ECS 实例和一台目标数据库 RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
相关文章
|
3月前
|
存储 NoSQL Redis
redis中Hash命令的基础操作
redis中Hash命令的基础操作
39 1
|
3月前
|
存储 NoSQL Serverless
Redis数据结构之——hash
Redis数据结构之——hash
|
28天前
|
存储 JSON NoSQL
Redis与Python的完美结合:实现高效数据交互和应用场景全解析
Redis与Python的完美结合:实现高效数据交互和应用场景全解析
110 0
|
1月前
|
存储 缓存 NoSQL
探索Redis的多样应用场景:加速和优化现代应用
探索Redis的多样应用场景:加速和优化现代应用
33 2
|
1月前
|
分布式计算 API 数据处理
Flink【基础知识 01】(简介+核心架构+分层API+集群架构+应用场景+特点优势)(一篇即可大概了解flink)
【2月更文挑战第15天】Flink【基础知识 01】(简介+核心架构+分层API+集群架构+应用场景+特点优势)(一篇即可大概了解flink)
55 1
|
1月前
|
存储 NoSQL Java
【Redis】1、学习 Redis 的五大基本数据类型【String、Hash、List、Set、SortedSet】
【Redis】1、学习 Redis 的五大基本数据类型【String、Hash、List、Set、SortedSet】
49 0
|
3月前
|
存储 NoSQL 关系型数据库
深度图解 Redis Hash(散列表)实现原理
深度图解 Redis Hash(散列表)实现原理
41 0
|
3月前
|
存储 NoSQL Java
深入了解 Redis Hash
深入了解 Redis Hash
26 0
|
3月前
|
NoSQL Java Redis
Redis的发布-订阅模式是什么?它的应用场景是什么?
Redis的发布-订阅模式是什么?它的应用场景是什么?
54 0
|
3月前
|
存储 NoSQL Redis
Redis(三)存储原理与数据模型(hash冲突、渐进式rehash)
Redis(三)存储原理与数据模型(hash冲突、渐进式rehash)
31 0