redis基本数据类型-hash(字典)

hash(字典)

Redis 的字典相当于 Java 语言里面的 HashMap,它是无序字典,内部存储了很多键值对。
在这里插入图片描述
Redis hash 是一个string类型的field和value的映射表,hash特别适合用于存储对象。

Redis 中每个 hash 可以存储 2^32 - 1 键值对(40多亿)。

实现结构上同 Java 的 HashMap 也是一致的,同样的数组 + 链表二维结构。第一维 hash 的数组位置碰撞 时,就会将碰撞的元素使用链表串接起来。
在这里插入图片描述

不同的是,Redis 的字典的值只能是字符串,另外它们 rehash 的方式不一样,因为 Java 的 HashMap 在字典很大时,rehash 是个耗时的操作,需要一次性全部 rehash。Redis 为了高性能,不能堵塞服务,所以采用了渐进式 rehash 策略。
在这里插入图片描述
渐进式 rehash 会在 rehash 的同时,保留新旧两个 hash 结构,查询时会同时查询两个 hash 结构,然后在后续的定时任务中以及 hash 的子指令中,循序渐进地将旧 hash 的内容 一点点迁移到新的 hash 结构中。

当 hash 移除了最后一个元素之后,该数据结构自动被删除,内存被回收。
hash 结构也可以用来存储用户信息,不同于字符串一次性需要全部序列化整个对象, hash 可以对用户结构中的每个字段单独存储。这样当我们需要获取用户信息时可以进行部分 获取。而以整个字符串的形式去保存用户信息的话就只能一次性全部读取,这样就会比较浪 费网络流量。
hash 也有缺点,hash 结构的存储消耗要高于单个字符串,到底该使用 hash 还是字符 串,需要根据实际情况再三权衡。

127.0.0.1:6379> hset books java "thinking in java" # hset为哈希表赋值,如果哈希表不存在,一个新的哈希表被创建并执行,如果已存在且字段也存在,覆盖旧值。命令行的字符串如果包含空格,要用引号括起来
(integer) 1
127.0.0.1:6379> hset books golang "concurrency in go"
(integer) 1
127.0.0.1:6379> hset books python "python cookbook"
(integer) 1
127.0.0.1:6379> hgetall books # hgetall返回哈希表中所有字段和值 key和value间隔出现
1) "java"
2) "thinking in java"
3) "golang"
4) "concurrency in go"
5) "python"
6) "python cookbook"
127.0.0.1:6379> hlen books # hlen获取字段数量
(integer) 3
127.0.0.1:6379> hget books java # hget返回哈希表中指定字段的值
"thinking in java"
127.0.0.1:6379> hset books golang "learning go programming" # 已有执行更新操作,所以返回0
(integer) 0
127.0.0.1:6379> hget books golang
"learning go programming"
127.0.0.1:6379> hmset books java "effective java" python "learning python" golang "modern golang programming" # 批量set
OK
127.0.0.1:6379> hgetall books
1) "java"
2) "effective java"
3) "golang"
4) "modern golang programming"
5) "python"
6) "learning python"
127.0.0.1:6379> hdel books golang # hdel删除key,都删除了books自动消除
(integer) 1
127.0.0.1:6379> hgetall books
1) "java"
2) "effective java"
3) "python"
4) "learning python"
127.0.0.1:6379> hmget books java python # hmget返回一个或多个给定字段的值
1) "effective java"
2) "learning python"
127.0.0.1:6379> hexists books java # hexists 查看哈希表的指定字段是否存在
(integer) 1
127.0.0.1:6379> hexists books c#
(integer) 0
// hincrby 命令用于为哈希表中的字段值加上指定增量值
127.0.0.1:6379> hset herobin age 23
(integer) 1
127.0.0.1:6379> hincrby herobin age 1
(integer) 24
// hvals 命令返回哈希表所有字段的值
127.0.0.1:6379> hvals books
1) "effective java"
2) "learning python"
// hincrbyfloat 命令用于为哈希表中的字段值加上指定浮点数增量值。
127.0.0.1:6379> hset herobin money 20.10
(integer) 0
127.0.0.1:6379> hincrbyfloat herobin money 0.1
"20.2"
// hkeys 命令用于获取哈希表中的所有字段名。
127.0.0.1:6379> hkeys books
1) "java"
2) "python"
// Hsetnx 命令用于为哈希表中不存在的的字段赋值,如果字段已经存在于哈希表中,操作无效。
127.0.0.1:6379> hsetnx herobin name zhangbin
(integer) 1
127.0.0.1:6379> hsetnx herobin name lisi
(integer) 0

hash常用方法

hset

Hset 命令用于为哈希表中的字段赋值 。
如果哈希表不存在,一个新的哈希表被创建并进行 HSET 操作。
如果字段已经存在于哈希表中,旧值将被覆盖。
HSET KEY_NAME FIELD VALUE
返回值
如果字段是哈希表中的一个新建字段,并且值设置成功,返回 1 。 如果哈希表中域字段已经存在且旧值已被新值覆盖,返回 0 。
实例

redis 127.0.0.1:6379> HSET myhash field1 "foo"
OK
redis 127.0.0.1:6379> HGET myhash field1
"foo"
redis 127.0.0.1:6379> HSET website google "www.g.cn"       # 设置一个新域
(integer) 1
redis 127.0.0.1:6379>HSET website google "www.google.com" # 覆盖一个旧域
(integer) 0

hget

Hget 命令用于返回哈希表中指定字段的值。
HGET KEY_NAME FIELD_NAME
返回值
返回给定字段的值。如果给定的字段或 key 不存在时,返回 nil 。
实例

# 字段存在
redis> HSET site redis redis.com
(integer) 1
redis> HGET site redis
"redis.com"
# 字段不存在 
redis> HGET site mysql
(nil)

hmset

Hmset 命令用于同时将多个 field-value (字段-值)对设置到哈希表中。
此命令会覆盖哈希表中已存在的字段。
如果哈希表不存在,会创建一个空哈希表,并执行 HMSET 操作。
HMSET KEY_NAME FIELD1 VALUE1 ...FIELDN VALUEN
返回值
如果命令执行成功,返回 OK 。
实例

redis 127.0.0.1:6379> HSET myhash field1 "foo" field2 "bar"
OK
redis 127.0.0.1:6379> HGET myhash field1
"foo"
redis 127.0.0.1:6379> HMGET myhash field2
"bar"

hmget

Hmget 命令用于返回哈希表中,一个或多个给定字段的值。
如果指定的字段不存在于哈希表,那么返回一个 nil 值。
HMGET KEY_NAME FIELD1...FIELDN
返回值
一个包含多个给定字段关联值的表,表值的排列顺序和指定字段的请求顺序一样。
实例

redis 127.0.0.1:6379> HSET myhash field1 "foo"
(integer) 1
redis 127.0.0.1:6379> HSET myhash field2 "bar"
(integer) 1
redis 127.0.0.1:6379> HMGET myhash field1 field2 nofield
1) "foo"
2) "bar"
3) (nil)

hgetall

Hgetall 命令用于返回哈希表中,所有的字段和值。
在返回值里,紧跟每个字段名(field name)之后是字段的值(value),所以返回值的长度是哈希表大小的两倍。
HGETALL KEY_NAME
返回值
以列表形式返回哈希表的字段及字段值。 若 key 不存在,返回空列表。
实例

redis 127.0.0.1:6379> HSET myhash field1 "foo"
(integer) 1
redis 127.0.0.1:6379> HSET myhash field2 "bar"
(integer) 1
redis 127.0.0.1:6379> HGETALL myhash
1) "field1"
2) "Hello"
3) "field2"
4) "World"

hexists

Hexists 命令用于查看哈希表的指定字段是否存在。
HEXISTS KEY_NAME FIELD_NAME
返回值
如果哈希表含有给定字段,返回 1 。 如果哈希表不含有给定字段,或 key 不存在,返回 0 。
实例

redis 127.0.0.1:6379> HSET myhash field1 "foo"
(integer) 1
redis 127.0.0.1:6379> HEXISTS myhash field1
(integer) 1
redis 127.0.0.1:6379> HEXISTS myhash field2
(integer) 0

hincrby

Hincrby 命令用于为哈希表中的字段值加上指定增量值。
增量也可以为负数,相当于对指定字段进行减法操作。
如果哈希表的 key 不存在,一个新的哈希表被创建并执行 HINCRBY 命令。
如果指定的字段不存在,那么在执行命令前,字段的值被初始化为 0 。
对一个储存字符串值的字段执行 HINCRBY 命令将造成一个错误。
本操作的值被限制在 64 位(bit)有符号数字表示之内。
HINCRBY KEY_NAME FIELD_NAME INCR_BY_NUMBER
返回值
执行 HINCRBY 命令之后,哈希表中字段的值。
实例

redis 127.0.0.1:6379> HSET myhash field1 20
(integer) 1
redis 127.0.0.1:6379> HINCRBY myhash field 1
(integer) 21
redis 127.0.0.1:6379> HINCRBY myhash field -1
(integer) 20

hlen

Hlen 命令用于获取哈希表中字段的数量。
HLEN KEY_NAME
返回值
哈希表中字段的数量。 当 key 不存在时,返回 0 。
实例

redis 127.0.0.1:6379> HSET myhash field1 "foo"
(integer) 1
redis 127.0.0.1:6379> HSET myhash field2 "bar"
(integer) 1
redis 127.0.0.1:6379> HLEN myhash
(integer) 2

hdel

Hdel 命令用于删除哈希表 key 中的一个或多个指定字段,不存在的字段将被忽略。
HDEL KEY_NAME FIELD1.. FIELDN
返回值
被成功删除字段的数量,不包括被忽略的字段。
实例

redis 127.0.0.1:6379> HSET myhash field1 "foo"
(integer) 1
redis 127.0.0.1:6379> HDEL myhash field1
(integer) 1
redis 127.0.0.1:6379> HDEL myhash field2
(integer) 0

hvals

Hvals 命令返回哈希表所有字段的值。
HVALS KEY_NAME FIELD VALUE
返回值
一个包含哈希表中所有值的表。 当 key 不存在时,返回一个空表。
实例

redis 127.0.0.1:6379> HSET myhash field1 "foo"
(integer) 1
redis 127.0.0.1:6379> HSET myhash field2 "bar"
(integer) 1
redis 127.0.0.1:6379> HVALS myhash
1) "foo"
2) "bar"
# 空哈希表/不存在的key
redis 127.0.0.1:6379> EXISTS not_exists
(integer) 0
redis 127.0.0.1:6379> HVALS not_exists
(empty list or set)

hincrbyfloat

Hincrbyfloat 命令用于为哈希表中的字段值加上指定浮点数增量值。
如果指定的字段不存在,那么在执行命令前,字段的值被初始化为 0 。
HINCRBYFLOAT KEY_NAME FIELD_NAME INCR_BY_NUMBER
返回值
执行 Hincrbyfloat 命令之后,哈希表中字段的值。
实例

redis 127.0.0.1:6379> HSET myhash field 20.50
(integer) 1
redis 127.0.0.1:6379> HINCRBYFLOAT mykey field 0.1
"20.60"

hkeys

Hkeys 命令用于获取哈希表中的所有字段名。
HKEYS KEY_NAME FIELD_NAME INCR_BY_NUMBER
返回值
包含哈希表中所有字段的列表。 当 key 不存在时,返回一个空列表。
实例

redis 127.0.0.1:6379> HSET myhash field1 "foo"
(integer) 1
redis 127.0.0.1:6379> HSET myhash field2 "bar"
(integer) 1
redis 127.0.0.1:6379> HKEYS myhash
1) "field1"
2) "field2"

hsetnx

Hsetnx 命令用于为哈希表中不存在的的字段赋值 。
如果哈希表不存在,一个新的哈希表被创建并进行 HSET 操作。
如果字段已经存在于哈希表中,操作无效。
如果 key 不存在,一个新哈希表被创建并执行 HSETNX 命令。
HSETNX KEY_NAME FIELD VALUE
返回值
设置成功,返回 1 。 如果给定字段已经存在且没有操作被执行,返回 0 。
实例

redis 127.0.0.1:6379> HSETNX myhash field1 "foo"
(integer) 1
redis 127.0.0.1:6379> HSETNX myhash field1 "bar"
(integer) 0
redis 127.0.0.1:6379> HGET myhash field1
"foo"
redis 127.0.0.1:6379> HSETNX nosql key-value-store redis
(integer) 1 
redis 127.0.0.1:6379> HSETNX nosql key-value-store redis       # 操作无效, key-value-store 已存在
(integer) 0
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值