Redis配置了redisTemplate.setEnableTransactionSupport(true)之后的坑,以及自己的理解

本文通过代码示例展示了在RedisTemplate启用事务支持后遇到的问题,包括乐观锁尝试与Redis事务结合时的冲突,以及在@Transactional注解方法内无法使用get或watch命令的情况。解析了Redis事务的工作原理,强调在启用事务时应避免直接使用get、watch、multi和exec命令,只需执行set操作。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

本文整合的redis客户端是lettucce,话不多说上代码。

先配置一个RedisTemplete的实例,在里面指定redis支持数据库的事务

@Bean
public RedisTemplate redisTemplate(LettuceConnectionFactory factory){
    RedisTemplate redisTemplate = new RedisTemplate();
    //使用LettuceConnectionFactory 代替 RedisConnectionFactory
    redisTemplate.setConnectionFactory(factory);
    //配置序列化方式
    redisTemplate.setKeySerializer(new StringRedisSerializer());
    redisTemplate.setValueSerializer(new GenericJackson2JsonRedisSerializer());
    redisTemplate.setHashKeySerializer(new StringRedisSerializer());
    redisTemplate.setHashValueSerializer(new GenericJackson2JsonRedisSerializer());

    //设置redis支持数据库的事务
    redisTemplate.setEnableTransactionSupport(true);

    return redisTemplate;
}

写个简单demo测试下事务是否有效

@Transactional(rollbackFor = Exception.class)
public void changeCount2(String name) throws Exception{
    redisTemplate.opsForValue().set(name,123);
    dotest();
    redisTemplate.opsForValue().set(name,345);
}<
对于同样的key我使用redis命令行能获取到她的值 但是用java代码却获取不到为什么? //用户信息 @ApiOperation("用户信息") @GetMapping("/info") public Rs login(HttpServletRequest request) { String token = request.getHeader("Authorization").substring(7); String key=RedisConst.USER_TOKEN + token; String userObj = (String) redisTemplate.boundValueOps(key).get(); User user = JSON.parseObject(userObj,User.class); if (user!=null){ System.out.println(user.getId()+" 、 "+user.getName()+" 、 "+user.getPassword() ); } return Rs.success(user); } package ys.config.redis; import com.fasterxml.jackson.annotation.JsonAutoDetect; import com.fasterxml.jackson.annotation.JsonTypeInfo; import com.fasterxml.jackson.annotation.PropertyAccessor; import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.jsontype.impl.LaissezFaireSubTypeValidator; import org.springframework.cache.CacheManager; import org.springframework.cache.interceptor.KeyGenerator; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.data.redis.cache.RedisCacheManager; import org.springframework.data.redis.connection.RedisConnectionFactory; import org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer; import org.springframework.data.redis.serializer.StringRedisSerializer; import javax.annotation.Resource; import java.lang.reflect.Method; import java.util.HashSet; import java.util.Set; @Configuration public class RedisConfig { @Resource private LettuceConnectionFactory lettuceConnectionFactory; @Bean public KeyGenerator keyGenerator() { return new KeyGenerator() { @Override public Object generate(Object target, Method method, Object... params) { StringBuffer sb = new StringBuffer(); sb.append(target.getClass().getName()); sb.append(method.getName()); for (Object obj : params) { sb.append(obj.toString()); } return sb.toString(); } }; } // 缓存管理器 @Bean public CacheManager cacheManager() { RedisCacheManager.RedisCacheManagerBuilder builder = RedisCacheManager.RedisCacheManagerBuilder .fromConnectionFactory(lettuceConnectionFactory); @SuppressWarnings("serial") Set<String> cacheNames = new HashSet<String>() { { add("codeNameCache"); } }; builder.initialCacheNames(cacheNames); return builder.build(); } /** * RedisTemplate配置 */ @Bean public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) { RedisTemplate<String, Object> template = new RedisTemplate<>(); template.setConnectionFactory(factory); // 使用 Jackson 序列化器 Jackson2JsonRedisSerializer<Object> serializer = new Jackson2JsonRedisSerializer<>(Object.class); template.setValueSerializer(serializer); template.setKeySerializer(new StringRedisSerializer()); template.afterPropertiesSet(); return template; }
最新发布
07-05
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值