12.3.2 使用 Redis Template
与其他的 Spring Data 项目类似,Spring Data Redis 以模板的形式提供了较高等级的数据访问方案。实际上,Spring Data Redis 提供了两个模板:
RedisTemplate
StringRedisTemplate
RedisTemplate 可以极大地简化 Redis 数据访问,能够让我们持久化各种类型的 key 和 value,并不局限于字节数组。在认识到 key 和 value 通常是 String 类型之后,StringRedisTemplate 扩展了 RedisTemplate,只关注 String 类型。
假设我们已经有了 RedisConnectionFactory,那么可以按照如下的方式构建 RedisTemplate:
注意,RedisTemplate 使用两个类型进行了参数化。第一个是 key 的类型,第二个是 value 的类型。在这里所构建的 RedisTemplate 中,将会保存 Product 对象作为 value,并将其赋予一个 String 类型的 key。
如果你所使用的 value 和 key 都是 String 类型,那么可以考虑使用 StringRedisTemplate 来代替 RedisTemplate:
注意,与 RedisTemplate 不同,StringRedisTemplate 有一个接受 RedisConnectionFactory 的构造器,因此没有必要在构建后再调用 setConnectionFactory()。
尽管这并非必须的,但是如果你经常使用 RedisTemplate 或 StringRedisTemplate 的话,你可以考虑将其配置为 bean,然后注入到需要的地方。如下就是一个声明 RedisTemplate 的简单 @Bean 方法:
如下是声明 StringRedisTemplate bean 的 @Bean 方法:
有了 RedisTemplate(或 StringRedisTemplate)之后,我们就可以开始保存、获取以及删除 key-value 条目了。RedisTemplate 的大多数操作都是表 12.5 中的子 API 提供的。
方法
子 API 接口
描述
opsForValue()
ValueOperations<K, V>
操作具有简单值的条目
opsForList()
ListOperations<K, V>
操作具有 list 值的条目
opsForSet()
SetOperations<K, V>
操作具有 set 值的条目
opsForZSet()
ZSetOperations<K, V>
操作具有 ZSet 值(排序的 set)的条目
opsForHash()
HashOperations<K, HK, HV>
操作具有 hash 值的条目
boundValueOps(K)
BoundValueOperations<K, V>
以绑定指定 key 的方式,操作具有简单值的条目
boundListOps(K)
BoundListOperations<K, V>
以绑定指定 key 的方式,操作具有 list 值的条目
boundSetOps(K)
BoundSetOperations<K, V>
以绑定指定 key 的方式,操作具有 set 值的条目
boundZSet(K)
BoundZSetOperations<K, V>
以绑定指定 key 的方式,操作具有 ZSet 值(排序的 set)的条目
boundHashOps(K)
BoundHashOperations<K, V>
以绑定指定 key 的方式,操作具有 hash 值的条目
我们可以看到,表 12.5 中的子 API 能够通过 RedisTemplate(和 StringRedisTemplate)进行调用。其中每个子 API 都提供了使用数据条目的操作,基于 value 中所包含的是单个值还是一个值的集合它们会有所差别。
这些子 API 中,包含了很多从 Redis 中存取数据的方法。我们没有足够的篇幅介绍所有的方法,但是会介绍一些最为常用的操作。
使用简单的值
假设我们想通过 RedisTemplate 保存 Product,其中 key 是 sku 属性的值。如下的代码片段展示了如何借助 opsForValue() 方法完成该功能:
类似地,如果你希望获取 sku 属性为 123456 的产品,那么可以使用如下的代码片段:
如果按照给定的 key,无法获得条目的话,将会返回 null。
使用 List 类型的值
使用 List 类型的 value 与之类似,只需使用 opsForList() 方法即可。例如,我们可以在一个 List 类型的条目尾部添加一个值:
通过这种方式,我们向列表的尾部添加了一个 Product,所使用的这个列表在存储时 key 为 cart。如果这个 key 尚未存在列表的话,将会创建一个。
rightPush() 会在列表的尾部添加一个元素,而 leftPush() 则会在列表的头部添加一个值:
我们有很多方式从列表中获取元素,可以通过 leftPop() 或 rightPop() 方法从列表中弹出一个元素:
除了从列表中获取值以外,这两个方法还有一个副作用就是从列表中移除所弹出的元素。如果你只是想获取值的话(甚至可能要在列表的中间获取),那么可以使用 range() 方法:
range() 方法不会从列表中移除任何元素,但是它会根据指定的 key 和索引范围,获取范围内的一个或多个值。前面的样例中,会获取 11 个元素,从索引为 2 的元素到索引为 12 的元素(不包含)。如果范围超出了列表的边界,那么只会返回索引在范围内的元素。如果该索引范围内没有元素的话,将会返回一个空的列表。
在 Set 上执行操作
除了操作列表以外,我们还可以使用 opsForSet() 操作 Set。最为常用的就是向其中添加一个元素:
在我们有多个 Set 并填充值之后,就可以对这些 Set 进行一些有意思的操作,如获取其差异、求交集和求并集:
当然,我们还可以移除它的元素:
我们甚至还可以随机获取 Set 中的一个元素:
因为 Set 没有索引和内部的排序,因此我们无法精准定位某个点,然后从 Set 中获取元素。
绑定到某个 key 上
表 12.5 包含了五个子 API,它们能够以绑定 key 的方式执行操作。这些子 API 与其他的 API 是对应的,但是关注于某一个给定的 key。
为了举例阐述这些子 API 的用法,我们假设将 Product 对象保存到一个 list 中,并且 key 为 cart。在这种场景下,假设我们想从 list 的右侧弹出一个元素,然后在 list 的尾部新增三个元素。我们此时可以使用 boundListOps() 方法所返回的 BoundListOperations:
注意,我们只在一个地方使用了条目的 key,也就是调用 boundListOps() 的时候。对返回的 BoundListOperations 执行的所有操作都会应用到这个 key 上。
Last updated