在redis中使用事务,通常的使用watch…multi…exec命令组合。
redis执行事务过程:
watch命令是可以监控redis的一些键;multi命令是开始事务,开始事务后,该客户端的命令不会马上被执行,而是存放到一个队列里,这时执行一些返回数据的命令,结果返回都是空。exec执行事务,在执行前判断watch监控的redis键的数据是否发生了变化(即使赋予之前相同的值也会被认为是变化过),如果它认为发生了变化,那么redis就会取消事务,否则执行事务。
@RequestMapping("/testwatch")
@ResponseBody
public void testwatch() {
redisTemplate.opsForValue().set("k1", "v1");
List list = (List) redisTemplate.execute(
(RedisOperations redisop) -> {
// 设置要监控的key值
redisop.watch("k1");
// 开启事务,在exec执行前,全部都只是进入队列
redisop.multi();
redisop.opsForValue().set("k2", "v2");
// 获取值将为空,因为redis只是把命令放入队列
Object value = redisop.opsForValue().get("k2");
// 此时手动更改k1的值
return redisop.exec();
}
);
System.out.println(list);
}
k1的值手动更改过,取消事务,Redis中k2的值未保存。
@RequestMapping("/testwatch")
@ResponseBody
public void testwatch() {
redisTemplate.opsForValue().set("k1", "v1");
List list = (List) redisTemplate.execute(
(RedisOperations redisop) -> {
// 设置要监控的key值
redisop.watch("k1");
// 开启事务,在exec执行前,全部都只是进入队列
redisop.multi();
redisop.opsForValue().set("k2", "v2");
// 因为k1是字符串,对字符串加一,显然是不能进行的
redisop.opsForValue().increment("k1", 1);
// 获取值将为空,因为redis只是把命令放入队列
Object value = redisop.opsForValue().get("k2");
return redisop.exec();
}
);
System.out.println(list);
}
k1是一个字符串,这里的代码对字符串加一,运行代码,可以看到服务器抛出了异常:“redis.clients.jedis.exceptions.JedisDataException: ERR value is not an integer or out of range”。
但是去查询k2的值,发现已经有了值。注意,这就是redis事务和数据库事务的不一样。
对应redis事务是先让命令进入队列,所以一开始他并没有检测这个“加一”是否能够成功,只是在exec执行时,才能发现错误,对于出错的命令redis只是报错,而错误后面的命令依旧被执行,所以k2有了值,这就是redis事务的特点,也是使用redis事务特别需要注意的地方。为了克服这个问题,一般我们在执行redis事务前,严格检查数据,以避免这样的情况发生。