Java Redis客户端概述
- Jedis是Redis的Java实现的客户端,提供了基本类型的支持,提供了比较全面的Redis命令的支持;了解Redis命令就能比较熟练的使用Jedis;
Jedis使用阻塞的I/O,且其方法调用都是同步的,程序流需要等到sockets处理完I/O才能执行,不支持异步。Jedis客户端实例不是线程安全的,所以需要通过连接池来使用Jedis。
- Redisson基于Netty框架来实现的,其方法调用是异步的。Redisson的API是线程安全的,所以可以操作单个Redisson连接来完成各种操作。实现了分布式和可扩展的Java数据结构,提供很多分布式相关操作服务,例如,分布式锁,分布式集合等
Redisson的宗旨是促进使用者对Redis的关注分离,从而让使用者能够将精力更集中地放在处理业务逻辑上。Redisson中的方法则是进行比较高的抽象,每个方法调用可能进行了一个或多个Redis方法调用。
- Lettuce:基于Netty框架的事件驱动的高级Redis客户端,用于线程安全同步,异步和响应使用,支持集群,Sentinel,管道和编码器。目前springboot默认使用的客户端。
Spring Data Redis
简述
目前的安装 redis版本 5.0.4,最新版本6.0;以下演示代码SpringBoot版本2.1.3;lettuce版本是5.1.4
Spring Data Redis目前支持两种驱动:Jedis和Lettuce,可以看成是这两种驱动的统一封装,以高度统一的形式屏蔽了底层驱动的操作细节,向用户提供一种统一的API
springboot 2.x版本中默认客户端是用lettuce实现的;在 springboot 1.5.x版本的默认的Redis客户端是 Jedis实现。
配置文件一览参考,注意网上很多都是spring.redis.jedis,不要随意引用,注意自己底层的实现
spring.redis.host=localhost
spring.redis.port=6379
spring.redis.password=root
## spring.redis.jedis和下面类似
# 连接池最大连接数(使用负值表示没有限制) 默认为8
spring.redis.lettuce.pool.max-active=8
# 连接池最大阻塞等待时间(使用负值表示没有限制) 默认为-1
spring.redis.lettuce.pool.max-wait=-1ms
# 连接池中的最大空闲连接 默认为8
spring.redis.lettuce.pool.max-idle=8
# 连接池中的最小空闲连接 默认为 0
spring.redis.lettuce.pool.min-idle=0
简单问题一览
Redis序列化报错
IllegalArgumentException: DefaultSerializer requires a Serializable payload but received
要缓存的 Java 对象必须实现 Serializable 接口,因为 Spring 会将对象先序列化再存入 Redis
Redis乱码
配置RedisTemplate序列化不然会乱码
注意:添加序列化之前的key,要屏蔽掉序列化才能找到;乱码要乱码的一样
RedisTemplate和StringRedisTemplate,配置就是全换成StringRedisSerializer
RedisTemplate使用的是 JdkSerializationRedisSerializer
StringRedisTemplate使用的是 StringRedisSerializer
代码演示
opsForValue - 操作 Key - Value ,简单的键值对
opsForList - 操作列表
opsForSet - 操作集合(set)
opsForZSet - 操作有序集合(sorted set,在 Redis 内叫 zSet)
opsForHash - 操作 Map 集合
opsForGeo - 操作地理信息,可以当作 HashSet 看,不过里面的 Key 是二维地理坐标
opsForHyperLogLog - 操作 HyperLogLog 值,是 counter 。
opsForCluster - 操作集群,管理集群用。
配置redisTemplate
@Bean
public RedisTemplate redisTemplate(RedisConnectionFactory factory) {
RedisTemplate<String, Object> template = new RedisTemplate<>();
template.setConnectionFactory(factory);
Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
ObjectMapper om = new ObjectMapper();
om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(om);
StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();
// key采用String的序列化方式
template.setKeySerializer(stringRedisSerializer);
// hash的key也采用String的序列化方式
template.setHashKeySerializer(stringRedisSerializer);
// value序列化方式采用jackson
template.setValueSerializer(jackson2JsonRedisSerializer);
// hash的value序列化方式采用jackson
template.setHashValueSerializer(jackson2JsonRedisSerializer);
template.afterPropertiesSet();
return template;
}
@Autowired
RedisTemplate redisTemplate;
@ApiOperation(value = "根据key(id)查询")
@RequestMapping(value = "/{key}", method = RequestMethod.GET)
public Object getById(@PathVariable("key") String key){
log.info("根据key {} 进行查询", key);
Person person = (Person)redisTemplate.opsForValue().get(key);
return person;
}
@ApiOperation("用id做key,保存")
@RequestMapping(value = "/save", method = RequestMethod.POST)
public void save(@RequestBody Person person){
log.info("调用保存接口");
//不存在就保存 存在就更新
redisTemplate.opsForValue().set(person.getId(), person);
}
@ApiOperation("删除")
@ApiImplicitParam(name = "key", value = "用户id")
@RequestMapping(value = "/{key}", method = RequestMethod.DELETE)
public Boolean delete(@PathVariable("key") String key){
log.info("调用删除接口");
// 不存在就返回false
return redisTemplate.delete(key);
}
@ApiOperation("更新")
@RequestMapping(value = "/update", method = RequestMethod.PUT)
public Object update(@ModelAttribute Person person){
//得到旧值 不存在就保存 存在就更新
Object andSet = redisTemplate.opsForValue().getAndSet(person.getId(), person);
return andSet;
}
set和list简单操作
@Autowired
RedisTemplate redisTemplate;
@ApiOperation(value = "多个value,分割,保存到set中")
@RequestMapping(value = "/saveToSet/{key}", method = RequestMethod.POST)
public Long save(@PathVariable("key") String key, @RequestParam String values){
log.info("保存values:{}", values);
// set保存,返回保存长度;重复项-1
return redisTemplate.opsForSet().add(key, values.split(","));
}
@ApiOperation(value = "将set中的值,移动到list中")
@RequestMapping(value = "/moveToList", method = RequestMethod.POST)
public long move(@RequestParam String setKey, @RequestParam String listKey){
log.info("将set{}中的值,移动到list{}中:", setKey, listKey);
Set members = redisTemplate.opsForSet().members(setKey);
log.info("将要copy的set长度为{},values{}", redisTemplate.opsForSet().size(setKey), members.toArray());
// 返回list长度
return redisTemplate.opsForList().leftPushAll(listKey, members);
}
@ApiOperation(value = "从list弹出一个值")
@ApiImplicitParams({
@ApiImplicitParam(name = "direction", value = "方向", dataType = "String", paramType = "query",
allowableValues = "left,right")
})
@RequestMapping(value = "/listPop", method = RequestMethod.GET)
public Object lkstPop(@RequestParam String listKey, @RequestParam String direction){
if(direction.equalsIgnoreCase("right")){
return redisTemplate.opsForList().rightPop(listKey);
}
return redisTemplate.opsForList().leftPop(listKey);
}
浅谈分布式锁
最早通过技术专题讨论第四期:漫谈分布式锁了解分布式锁的概念;简单理解的话,就是需要一个分布式情况下的协调器(对外是相对单点的),那我们常用的选项就有以下几项:
- 基于数据库
在数据库中创建一张表,表里包含方法名等字段,并且在方法名字段上面创建唯一索引,执行某个方法需要使用此方法名向表中插入数据,成功插入则获取锁,执行结束则删除对应的行数据释放锁
- 基于缓存数据库Redis
Redis性能好并且实现方便,但是单节点的分布式锁在故障迁移时产生安全问题,Redlock是Redis的作者 Antirez 提出的集群模式分布式锁,基于N个完全独立的Redis节点实现分布式锁的高可用
- 基于ZooKeeper
ZooKeeper 是以 Paxos 算法为基础的分布式应用程序协调服务,为分布式应用提供一致性服务的开源组件
Redis分布式锁
简单分布式锁
最初分布式锁借助于setnx和expire命令,但是这两个命令不是原子操作,如果执行 setnx之后获取锁但是此时客户端挂掉,这样无法执行expire设置过期时间就导致锁 一直无法被释放,因此在2.8版本中Antirez为setnx增加了参数扩展, 使得setnx和expire具备原子操作性。
方法Boolean setIfAbsent(K key, V value, long timeout, TimeUnit unit);是在Spring Data Redis2.1版本中新增的;
加锁和续期:加锁使用测试2;续期使用测试3
@ApiOperation("测试1")
@RequestMapping(value = "/test3", method = RequestMethod.GET)
public boolean test3(@ModelAttribute Person person){
// 设置过期时间 10秒/ 不存在返回false
return redisTemplate.expire(person.getName(), 10, TimeUnit.SECONDS);
}
@ApiOperation("测试2,不存在key就新建并设置过期时间")
@RequestMapping(value = "/test", method = RequestMethod.GET)
public Boolean test(@ModelAttribute Person person){
//test:张三 文件夹
return redisTemplate.opsForValue().setIfAbsent(person.getName(), person, 10, TimeUnit.SECONDS);
}
@ApiOperation("测试3,存在就更新并设置过期时间,存在返回true,不存在返回false")
@RequestMapping(value = "/test2", method = RequestMethod.GET)
public Boolean test2(@ModelAttribute Person person){
return redisTemplate.opsForValue().setIfPresent(person.getName(), person, 10, TimeUnit.SECONDS);
}
待优化
续期优化:可以在事务中check锁是你的锁,然后再续期,不要瞎续期;或者用redis lua脚本
解锁优化:同理检查锁是你的锁
Redis集群和RedLock算法
如果在分布式环境中,线程A设置锁后,master节点挂了,但是锁没有同步给slave,然后slave升级成master节点,这时候线程B来设置锁,导致线程A和B同时拥有锁,由此产生了RedLock算法;
在Redis的分布式环境中,我们假设有N个完全互相独立的Redis节点,在N个Redis实例上使用与在Redis单实例下相同方法获取锁和释放锁。
现在假设有5个Redis主节点(大于3的奇数个),这样基本保证他们不会同时都宕掉,获取锁和释放锁的过程中,客户端会执行以下操作:
1.获取当前Unix时间,以毫秒为单位
2.依次尝试从5个实例,使用相同的key和具有唯一性的value获取锁
当向Redis请求获取锁时,客户端应该设置一个网络连接和响应超时时间,这个超时时间应该小于锁的失效时间,这样可以避免客户端死等
3.客户端使用当前时间减去开始获取锁时间就得到获取锁使用的时间。当且仅当从半数以上的Redis节点取到锁,并且使用的时间小于锁失效时间时,锁才算获取成功
4.如果取到了锁,key的真正有效时间等于有效时间减去获取锁所使用的时间,这个很重要
5.如果因为某些原因,获取锁失败(没有在半数以上实例取到锁或者取锁时间已经超过了有效时间),客户端应该在所有的Redis实例上进行解锁,无论Redis实例是否加锁成功,因为可能服务端响应消息丢失了但是实际成功了,毕竟多释放一次也不会有问题
当然有人也对这个算法提出了质疑,这个问题仁者见仁智者见智;
Java的redLock实现包是Redisson,可以使用Redisson直接来使用集群的分布式锁。