赞
踩
由于Redis官方文档的编写水平实在是难以恭维,因此本文翻译难度极大。阅读本文前须知:
Redis的字符串存储字节序列,包括文本、序列化对象和二进制数组。因此,字符串是您可以与Redis键关联的最简单类型的值。它们通常用于缓存,但它们还支持额外的功能,使您能够实现计数器并执行位操作。
由于Redis键是字符串,当我们将字符串类型用作值时,实际上是将一个字符串映射到另一个字符串。
> SET bike:1 Deimos
OK
> GET bike:1
"Deimos"
SET
命令具有一些有趣的选项,这些选项作为附加参数提供。例如,我可以要求SET
在键已经存在时失败,或者相反,只有在键已经存在时才成功:
> set bike:1 bike nx
(nil)
> set bike:1 bike xx
OK
在单个命令中设置或检索多个键的值的能力对于降低延迟也很有用。因此,有MSET
和MGET
命令:
> mset bike:1 "Deimos" bike:2 "Ares" bike:3 "Vanth"
OK
> mget bike:1 bike:2 bike:3
1) "Deimos"
2) "Ares"
3) "Vanth"
当使用MGET
时,Redis返回一个值数组。
即使字符串是Redis的基本值,您仍然可以执行一些有趣的操作。例如,其中之一是原子递增:
> set total_crashes 0
OK
> incr total_crashes
(integer) 1
> incrby total_crashes 10
(integer) 11
INCR
命令将字符串值解析为整数,将其递增1,最后将得到的值设置为新值。还有其他类似的命令,如INCRBY
、DECR
和DECRBY
。
什么是原子递增的含义呢?即使多个客户端针对相同的键发出INCR
命令,它们也永远不会陷入竞态条件。
默认情况下,单个Redis字符串的最大容量为512MB。
Redis lists是字符串值的链表,经常用于:
LPUSH
命令在 Redis 中将新元素添加到列表的左侧(头部),而 RPUSH
命令将新元素添加到列表的右侧(尾部)。最后,LRANGE
命令从列表中提取元素的范围:
> RPUSH bikes:repairs bike:1
(integer) 1
> RPUSH bikes:repairs bike:2
(integer) 2
> LPUSH bikes:repairs bike:important_bike
(integer) 3
> LRANGE bikes:repairs 0 -1
1) "bike:important_bike"
2) "bike:1"
3) "bike:2"
在Redis列表上定义的一个重要操作是弹出元素。弹出元素是从列表中检索元素并同时将其从列表中删除的操作。您可以从左侧和右侧弹出元素,类似于您可以在列表的两侧推送元素。我们将添加三个元素并弹出三个元素,因此在这系列命令结束时,列表将为空,不再有元素可以弹出:
> RPUSH bikes:repairs bike:1 bike:2 bike:3
(integer) 3
> RPOP bikes:repairs
"bike:3"
> LPOP bikes:repairs
"bike:1"
> RPOP bikes:repairs
"bike:2"
> RPOP bikes:repairs
(nil)
Redis返回一个NULL
值,表示列表中没有元素。
在许多用例中,我们只希望使用列表存储最新的项目,无论这些项目是什么:社交网络更新、日志或其他任何内容。
Redis 允许我们将列表用作有上限的集合,仅记住最新的 N 个项目,并使用 LTRIM
命令丢弃所有最旧的项目。
LTRIM
命令类似于 LRANGE
,但它不是显示指定范围的元素,而是将此范围设置为新的列表值。给定范围之外的所有元素都将被移除。
例如,如果您将自行车添加到维修列表的末尾,但只关心列表中最久的前 3 辆自行车:
> RPUSH bikes:repairs bike:1 bike:2 bike:3 bike:4 bike:5
(integer) 5
> LTRIM bikes:repairs 0 2
OK
> LRANGE bikes:repairs 0 -1
1) "bike:1"
2) "bike:2"
3) "bike:3"
列表具有一项特殊功能,使它们适用于实现队列,并且通常作为进程间通信系统的构建块:阻塞操作。
假设您希望使用一个进程将项目推送到列表中,并使用另一个进程实际处理这些项目。这是通常的生产者/消费者设置,可以以以下简单的方式实现:
LPUSH
。RPOP
。然而,有时可能列表是空的,没有可处理的内容,因此 RPOP
只会返回 NULL
。在这种情况下,消费者被迫等待一段时间,然后使用 RPOP
再次重试。这被称为轮询,但在这个上下文中不是一个好主意,因为它有一些缺点:
NULL
)。NULL
后,它会等待一段时间。为了使延迟更小,我们可以在调用 RPOP
之间等待时间较短,这会放大问题 1,即对 Redis 的更多无用调用。因此,Redis 实现了名为 BRPOP
和 BLPOP
的命令,它们是 RPOP
和 LPOP
的版本,如果列表为空,它们将阻塞:它们只在向列表添加新元素时或达到用户指定的超时时才向调用者返回。
请注意,您可以使用 0 作为超时,永远等待元素,并且还可以指定多个列表,而不仅仅是一个,以便同时等待多个列表,并在第一个列表接收到元素时得到通知。
关于 BRPOP
的一些要点:
RPOP
不同:它是一个包含两个元素的数组,因为 BRPOP
和 BLPOP
还包括键的名称,因为它们能够等待来自多个列表的元素。NULL
。Redis会在列表为空时自动删除键,或者在尝试向空列表中添加元素时创建空列表。这适用于所有由多个元素组成的Redis数据类型。
这不仅适用于列表,还适用于由多个元素组成的所有 Redis 数据类型,包括 Streams、Sets、Sorted Sets 和 Hashes。
基本上,我们可以用三条规则总结这种行为:
LLEN
命令)或删除元素的写命令,对于一个空键,总是产生与键持有的聚合类型为空相同的结果,该类型是命令期望找到的类型。Redis列表的最大长度是 2 32 − 1 2^{32} - 1 232−1个元素。
Set是一组无序的唯一字符串。您可以使用Set来高效地:
SADD
命令向集合添加新元素。还可以对集合执行许多其他操作,例如测试给定元素是否已存在,执行多个集合之间的交集、并集或差集等。
> SADD bikes:racing:france bike:1 bike:2 bike:3
(integer) 3
> SMEMBERS bikes:racing:france
1) bike:3
2) bike:1
3) bike:2
Redis 提供了用于测试集合成员身份的命令。这些命令可以用于单个项目以及多个项目:
> SISMEMBER bikes:racing:france bike:1
(integer) 1
> SMISMEMBER bikes:racing:france bike:2 bike:3 bike:4
1) (integer) 1
2) (integer) 1
3) (integer) 0
我们还可以找到两个集合之间的差异。例如,我们可能想知道哪些自行车在法国比赛,但不在美国比赛:
> SADD bikes:racing:usa bike:1 bike:4
(integer) 2
> SDIFF bikes:racing:france bikes:racing:usa
1) "bike:3"
2) "bike:2"
还有其他一些不太平凡的操作,仍然可以使用正确的 Redis 命令轻松实现。例如,我们可能想要列出在法国、美国和其他一些比赛中参赛的所有自行车。我们可以使用 SINTER
命令来执行不同集合之间的交集。除了交集之外,您还可以执行并集、差集等操作。例如,如果我们添加了第三场比赛,我们可以看到其中一些命令的实际效果:
> SADD bikes:racing:france bike:1 bike:2 bike:3 (integer) 3 > SADD bikes:racing:usa bike:1 bike:4 (integer) 2 > SADD bikes:racing:italy bike:1 bike:2 bike:3 bike:4 (integer) 4 > SINTER bikes:racing:france bikes:racing:usa bikes:racing:italy 1) "bike:1" > SUNION bikes:racing:france bikes:racing:usa bikes:racing:italy 1) "bike:2" 2) "bike:1" 3) "bike:4" 4) "bike:3" > SDIFF bikes:racing:france bikes:racing:usa bikes:racing:italy (empty array) > SDIFF bikes:racing:france bikes:racing:usa 1) "bike:3" 2) "bike:2" > SDIFF bikes:racing:usa bikes:racing:france 1) "bike:4"
请注意,当所有集合之间的差异为空时,SDIFF
命令返回一个空数组。还要注意,传递给 SDIFF
的集合顺序很重要,因为差异不是可交换的。
当您想要从集合中删除项目时,您可以使用 SREM
命令从集合中移除一个或多个项目,或者您可以使用 SPOP
命令从集合中随机移除一个项目。您还可以使用 SRANDMEMBER
命令从集合中返回一个随机项目,而不将其删除:
> SADD bikes:racing:france bike:1 bike:2 bike:3 bike:4 bike:5
(integer) 5
> SREM bikes:racing:france bike:1
(integer) 1
> SPOP bikes:racing:france
"bike:3"
> SMEMBERS bikes:racing:france
1) "bike:2"
2) "bike:4"
3) "bike:5"
> SRANDMEMBER bikes:racing:france
"bike:2"
Redis集合的最大大小是 2 32 − 1 个 2^{32} - 1个 232−1个成员。
Redis 哈希是一种记录类型,结构化为键值对的集合。您可以使用哈希来表示基本对象,以及存储计数器等内容的分组。
> HSET bike:1 model Deimos brand Ergonom type 'Enduro bikes' price 4972
(integer) 4
> HGET bike:1 model
"Deimos"
> HGET bike:1 price
"4972"
> HGETALL bike:1
1) "model"
2) "Deimos"
3) "brand"
4) "Ergonom"
5) "type"
6) "Enduro bikes"
7) "price"
8) "4972"
HSET
命令设置哈希的多个字段,而 HGET
检索单个字段。HMGET
类似于 HGET
,但返回一个值数组:
> HMGET bike:1 model price no-such-field
1) "Deimos"
2) "4972"
3) (nil)
有些命令也能在单个字段上执行操作,比如 HINCRBY
:
> HINCRBY bike:1 price 100
(integer) 5072
> HINCRBY bike:1 price -100
(integer) 4972
每个哈希可以存储多达 2 32 − 1 2^{32} - 1 232−1个键值对。实际上,您的哈希受到托管 Redis 部署的 VM 上总内存的限制。
Redis 有序集合是由关联分数排序的一组唯一字符串的集合。当多个字符串具有相同的分数时,字符串按词典顺序排序。一些有序集合的用例包括:
有序集合通过一个双端口数据结构实现,包含跳表和哈希表,因此当我们要求排序元素时,Redis 实际上根本不需要做任何工作,它已经排序好了。
让我们从一个简单的例子开始,将所有参赛者及他们在第一场比赛中获得的分数添加到有序集合中:
> ZADD racer_scores 10 "Norem"
(integer) 1
> ZADD racer_scores 12 "Castilla"
(integer) 1
> ZADD racer_scores 8 "Sam-Bodden" 10 "Royce" 6 "Ford" 14 "Prickett"
(integer) 4
正如您所见,ZADD
与 SADD
类似,但多了一个额外的参数,即分数。ZADD
也是可变参数的,因此您可以自由指定多个分数-值对。
让我们获取所有得分为 10 分或更少的赛车手。我们使用 ZRANGEBYSCORE
命令来完成:
> ZRANGEBYSCORE racer_scores -inf 10
1) "Ford"
2) "Sam-Bodden"
3) "Norem"
4) "Royce"
要删除一个元素,我们只需调用 ZREM
命令并提供赛车手的姓名。还可以删除一系列元素。让我们删除 Castilla 赛车手以及所有分数严格少于 10 分的赛车手:
> ZREM racer_scores "Castilla"
(integer) 1
> ZREMRANGEBYSCORE racer_scores -inf 9
(integer) 2
> ZRANGE racer_scores 0 -1
1) "Norem"
2) "Royce"
3) "Prickett"
对于有序集合元素,另一个非常有用的操作是获取排名。还可以使用 ZREVRANK
命令,以便按降序排列的方式获取排名:
> ZRANK racer_scores "Norem"
(integer) 0
> ZREVRANK racer_scores "Norem"
(integer) 3
假设有序集合中的元素都以相同的分数插入,Redis允许按字典顺序获取范围。操作词典范围的主要命令包括 ZRANGEBYLEX
、ZREVRANGEBYLEX
、ZREMRANGEBYLEX
和 ZLEXCOUNT
。
这个特性很重要,因为它允许我们将有序集合用作通用索引。例如,如果要按照 128 位无符号整数参数对元素进行索引,您只需将元素添加到具有相同分数的有序集合中,但带有一个由 128 位数字组成的大端字节序的 16 字节前缀。由于大端序的数字,在字典顺序排序时实际上也是按数值排序的,所以您可以在 128 位空间中请求范围,并且获取元素的值时舍弃前缀。
可以对已包含在有序集合中的元素调用 ZADD
来更新它们的分数。
由于这个特性,常见的用例是排行榜。典型的应用是 Facebook 游戏,其中您可以将按高分对用户进行排序的能力与获取排名操作结合使用,以显示前 N 个用户和用户在排行榜中的排名。
Redis Stack的JSON功能为Redis提供了对JavaScript对象表示法(JSON)的支持。它使您能够在Redis数据库中存储、更新和检索JSON值,就像处理其他Redis数据类型一样。Redis JSON还可以与搜索和查询无缝集成,使您能够有效地对JSON文档进行索引和查询。
第一个要尝试的JSON命令是JSON.SET
,它使用JSON值设置Redis键。JSON.SET
接受所有JSON值类型。以下是一个创建JSON字符串的示例:
> JSON.SET animal $ '"dog"'
"OK"
> JSON.GET animal $
"[\"dog\"]"
> JSON.TYPE animal $
1) "string"
请注意,命令中包含的美元符号字符$
是JSON文档中值的路径(在这种情况下,它只是表示根)。
JSON.DEL
命令使用路径参数删除指定的任何JSON值:
> JSON.SET example $ '[ true, { "answer": 42 }, null ]'
OK
> JSON.GET example $
"[[true,{\"answer\":42},null]]"
> JSON.GET example $[1].answer
"[42]"
> JSON.DEL example $[-1]
(integer) 1
> JSON.GET example $
"[[true,{\"answer\":42}]]"
您可以使用一组专用的JSON命令来操作数组:
> JSON.SET arr $ [] OK > JSON.ARRAPPEND arr $ 0 1) (integer) 1 > JSON.GET arr $ "[[0]]" > JSON.ARRINSERT arr $ 0 -2 -1 1) (integer) 3 > JSON.GET arr $ "[[-2,-1,0]]" > JSON.ARRTRIM arr $ 1 1 1) (integer) 1 > JSON.GET arr $ "[[-1]]" > JSON.ARRPOP arr $ 1) "-1" > JSON.ARRPOP arr $ 1) (nil)
JSON对象也有自己的命令:
> JSON.SET obj $ '{"name":"Leonard Cohen","lastSeen":1478476800,"loggedOut": true}'
OK
> JSON.OBJLEN obj $
1) (integer) 3
> JSON.OBJKEYS obj $
1) 1) "name"
2) "lastSeen"
3) "loggedOut"
路径帮助您在 JSON 文档中访问特定的元素。由于没有标准的 JSON 路径语法,Redis JSON 实现了自己的路径语法。JSON 的语法基于常见的最佳实践,且类似于 JSONPath。
返回的值是一个 JSON 字符串,其中包含顶级数组,数组中的元素是 JSON 序列化后的字符串。如果使用了多个路径,则返回值是一个 JSON 字符串,其中包含顶级对象,对象的值是序列化后的 JSON 值数组。
以下 JSONPath 语法表格是根据 Goessner 的路径语法比较进行调整的:
语法元素 | 描述 |
---|---|
$ | 根,最外层的JSON元素 |
. or [] | 选择子元素 |
.. | 递归地遍历JSON文档 |
* | 通配符 |
[] | 下标操作符,访问数组元素 |
[,] | 联合,选择多个元素 |
[start 声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Li_阴宅/article/detail/748261 推荐阅读 相关标签 Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。 |