一、哈希(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类型
- 由于博主才疏学浅,难免会有纰漏,假如你发现了错误或偏见的地方,还望留言给我指出来,我会对其加以修正。
- 如果你觉得文章还不错,你的转发、分享、点赞、留言就是对我最大的鼓励。
- 感谢您的阅读,十分欢迎并感谢您的关注。