场景:需要存两条数据到redis中,并且两条要么都存要么都不存,需要事务来控制
Spring Data Redis的RedisTemplate提供了MULTI、EXEC命令进行封装,远看可以解决问题,代码实现:
redisOperations.multi();
redisOperations.opsForHash().put("xxx", field, hashValue);
redisOperations.opsForSet().add(key,setvalue);
redisOperations.exec();
但是执行之后发现redis中并没有存入想要的数据,而且抛出异常 No ongoing transaction. Did you forget to call multi?
原因:发现上面代码中mutil()方法和exec()方法都会从新建立新的连接,导致数据丢失
解决方法:采用RedisTemplate的SesionCallback实现在同一个Connection中,完成多个操作的方法:
SessionCallback sessionCallback = new SessionCallback() {
@Override
public Object execute(RedisOperations redisOperations) throws DataAccessException {
redisOperations.multi();
redisOperations.opsForHash().put("xxx", field, hashValue);
redisOperations.opsForSet().add(key,setvalue);
return redisOperations.exec();
}
};
redisTemplate.execute(sessionCallback);
注意:匿名内部类中要使用外部类的局部变量,局部变量需要使用final来修饰。
原因:局部变量的生命周期:当该方法被调用时,该方法中的局部变量在栈中被创建,当方法调用结束时,退栈,这些局部变量全部死亡。而内部类对象生命周期与其它类一样:自创建一个匿名内部类对象,系统为该对象分配内存,直到没有引用变量指向分配给该对象的内存,它才会死亡(被JVM垃圾回收)。所以完全可能出现的一种情况是:成员方法已调用结束,局部变量已死亡,但匿名内部类的对象仍然活着。所以如果内部类对象访问了同一个方法的局部变量就要求该局部变量的生命周期不能小于匿名类的生命周期,所以需要将被匿名内部类使用到的局部变量使用final修饰。定义为final后,编译程序的实现方法:对于匿名内部类对象要访问的所有final类型局部变量,都拷贝成为该对象中的一个数据成员。这样,即使栈中局部变量已死亡,但被定义为final类型的局部变量的值永远不变,因而匿名内部类对象在局部变量死亡后,照样可以访问final类型的局部变量,因为它自己拷贝了一份,且与原局部变量的值始终一致。