redis发生死锁问题怎么办

71次阅读
没有评论

共计 2014 个字符,预计需要花费 6 分钟才能阅读完成。

自动写代码机器人,免费开通

这篇文章主要介绍了 redis 发生死锁问题怎么办,具有一定借鉴价值,需要的朋友可以参考下。希望大家阅读完这篇文章后大有收获。下面让丸趣 TV 小编带着大家一起了解一下。

就分布式锁而言,一个常用的问题就是如果一个服务 setnx 成功了,但是在解锁的时候如果发生了宕机或者一些特殊因素,导致无法解锁,那么其他服务将陷入死锁的状态。所以,我们在用 setnx 的同时想着去用 expire 指令对锁进行一个过期操作, 从指令可以看出 setnx 和 expire 指令是分开的,如果在这中间的空隙过程中如果有特殊因素导致指令无法继续,也会导致死锁的产生。

解决方法:

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Component;
import org.springframework.util.StringUtils;
@Component
public class RedisLock {Logger logger = LoggerFactory.getLogger(this.getClass());
 @Autowired
 private StringRedisTemplate redisTemplate;
 * 加锁
 * @param key 
 * @param value 当前时间 + 超时时间
 * @return
 public boolean lock(String key, String value) {if (redisTemplate.opsForValue().setIfAbsent(key, value)) { 
 // 这个其实就是 setnx 命令,只不过在 java 这边稍有变化,返回的是 boolean
 // 设置个过期时间,当然如果在这中间的空隙过程中如果有特殊因素导致指令无法继续,也会导致死锁的产生,如果死锁出现,则后续代码会处理
 redisTemplate.expire(key, lockTime, TimeUnit.SECONDS);
 return true;
 // 避免死锁,且只让一个线程拿到锁
 String currentValue = redisTemplate.opsForValue().get(key);
 // 如果锁过期了
 if (!StringUtils.isEmpty(currentValue) Long.parseLong(currentValue) System.currentTimeMillis()) {
 // 获取上一个锁的时间
 String oldValues = redisTemplate.opsForValue().getAndSet(key, value);
 只会让一个线程拿到锁
 如果旧的 value 和 currentValue 相等,只会有一个线程达成条件,因为第二个线程拿到的 oldValue 已经和 currentValue 不一样了
 if (!StringUtils.isEmpty(oldValues) oldValues.equals(currentValue)) {
 return true;
 return false;

String currentValue = redisTemplate.opsForValue().get(key); if (!StringUtils.isEmpty(currentValue) currentValue.equals(value)) {redisTemplate.opsForValue().getOperations().delete(key); } catch (Exception e) {logger.error( redis 分布式锁解锁异常,{} , e); }

调用:

// 加锁
 long time = System.currentTimeMillis() + 1000 * lockTime // 超时时间:10 秒,最好设为常量
 boolean isLock = redisLock.lock(...keyName, String.valueOf(time));
 if(!isLock){
 throw new RuntimeException( 系统正忙 
 // doSomething...

// 解锁 redisLock.unlock(...keyName, String.valueOf(time));

感谢你能够认真阅读完这篇文章,希望丸趣 TV 小编分享 redis 发生死锁问题怎么办内容对大家有帮助,同时也希望大家多多支持丸趣 TV,关注丸趣 TV 行业资讯频道,遇到问题就找丸趣 TV,详细的解决方法等着你来学习!

向 AI 问一下细节

丸趣 TV 网 – 提供最优质的资源集合!

正文完
 
丸趣
版权声明:本站原创文章,由 丸趣 2024-02-03发表,共计2014字。
转载说明:除特殊说明外本站除技术相关以外文章皆由网络搜集发布,转载请注明出处。
评论(没有评论)