Redisson 使用分布式锁出现 WRONGTYPE Operation against a key holding the wrong kind of value 错误

异常信息

org.redisson.client.RedisException: WRONGTYPE Operation against a key holding the wrong kind of value.

结论

  • 分布式锁使用的key和别的键值对共用了同一个键

为什么这么说呢,写一段junit测试代码,这里RedissonClient配置就不展示了,有兴趣可以自行去了解一下

@RunWith(SpringRunner.class)
@SpringBootTest(classes = AccountServiceApplication.class)
class IScMemberProfitImplServiceTest {

	@Autowired
    private RedissonClient redissonClient;

	@Test
    void testRedisson() {
        //获取12345678对应的键
        RBucket<Object> bucket = redissonClient.getBucket("12345678");
        //获取键对应的值(此时为null)
        String o1 = (String)bucket.get();
        //设置同样的key
        RLock lock = redissonClient.getLock("12345678");
        //获取分布式锁,有效期2分钟
        lock.lock(2, TimeUnit.MINUTES);
        //测试重入锁
        //lock.lock(2, TimeUnit.MINUTES);
        //lock.lock(2, TimeUnit.MINUTES);

		//设置这个键对应的值为999,有效期为两分钟
        bucket.set("999",2, TimeUnit.MINUTES);
        
 		//释放分布式锁
        if(lock.isLocked()){
        	if(lock.isHeldByCurrentThread()){
            	lock.unlock();
            }
        }
}

运行完以上代码,线程在 if(lock.isHeldByCurrentThread()){ 处抛出如题异常,

回看以上代码,在分布式锁获取后,查看redis看到,redis中‘12345678’对应存的hash类型,内容如下
在这里插入图片描述
那运行到bucket.set("999",2, TimeUnit.MINUTES); 为什么不会报错呢? 分布式锁存储了hash类型,这不跟String类型冲突了吗

emmm,这要是在mysql确实执行不过去,但对于redis来说,set就意味着直接覆盖,因此,原来存储的不管是什么类型,都会直接覆盖


那为什么执行到if(lock.isHeldByCurrentThread()){ 这里又会报错呢?

额,接着上面所说,由于键 “12345678” 对应的值呗覆盖成了 “999” 字符串,线程执行到此行代码是,判断当前执行的线程是否与redis中存储的线程信息一致,但redis中的值结构已经不再是hash类型,因此会抛出异常

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用Redisson实现分布式锁可以确保在分布式环境中对共享资源的访问是互斥的。下面是使用Redisson实现分布式锁的步骤: 1. 添加Redisson依赖:在`pom.xml`文件中添加Redisson依赖。 ```xml <dependency> <groupId>org.redisson</groupId> <artifactId>redisson</artifactId> <version>3.15.5</version> </dependency> ``` 2. 配置Redisson连接信息:在`application.properties`文件中配置Redis连接信息。 ```properties spring.redis.host=127.0.0.1 spring.redis.port=6379 ``` 3. 创建RedissonClient Bean:在配置类中创建RedissonClient Bean,用于获取分布式锁实例。 ```java @Configuration public class RedissonConfig { @Value("${spring.redis.host}") private String redisHost; @Value("${spring.redis.port}") private String redisPort; @Bean(destroyMethod = "shutdown") public RedissonClient redissonClient() { Config config = new Config(); config.useSingleServer().setAddress("redis://" + redisHost + ":" + redisPort); return Redisson.create(config); } } ``` 4. 使用分布式锁:在需要进行互斥访问的代码块中使用分布式锁。 ```java @Autowired private RedissonClient redissonClient; public void doSomething() { RLock lock = redissonClient.getLock("myLock"); try { // 尝试获取锁,等待10秒,锁自动释放时间为30秒 boolean isLocked = lock.tryLock(10, 30, TimeUnit.SECONDS); if (isLocked) { // 获得锁之后执行业务逻辑 // ... } else { // 获取锁失败,处理异常情况 // ... } } catch (InterruptedException e) { // 处理异常 // ... } finally { // 释放锁 lock.unlock(); } } ``` 在以上示例中,我们使用Redisson的RLock对象来获取分布式锁使用`tryLock`方法可以尝试获取锁,如果获取成功,则执行相应的业务逻辑;如果获取失败,则根据实际情况处理异常。最后,使用`unlock`方法释放锁。 通过以上步骤,就可以使用Redisson实现分布式锁,确保在分布式环境中对共享资源的互斥访问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值