使用hiredis实现redis分布式锁

简言

1. redis实现分布式锁的原理,这里不再赘述,不清楚的可以参见笔者的这篇博客

https://blog.csdn.net/yzf279533105/article/details/100524700

2. 解锁时使用lua脚本,由于hiredis是根据空格来解析cmd参数的,但是lua中肯定有空格,所以解锁的redis命令要分开格式化

3. 重点关注 Unlock()函数即可,关注里面如何调用lua脚本的

 

代码如下:

头文件(仅贴出主要代码)

// redis锁
struct RedisLock
{
	string 		key;		// 锁的key
	int 		randval;	// 随机值
};


// 	加锁(锁数据,过期时间,单位:秒)
bool Lock(RedisLock lockkey, uint32_t expire);

// 解锁
bool Unlock(RedisLock lockkey);

cpp文件

// 	加锁(锁数据,过期时间,单位:秒)
bool CRedisClient::Lock(RedisLock lockkey, uint32_t expire)
{
	bool bSuc = connect();
	if (!bSuc) 
	{
		ERROR("CRedisClient::Lock(), connect failed");
		return false;
	}

	ostringstream os;
	os<< "set " << lockkey.key <<" "<< lockkey.randval << " ex " << expire << "  nx";

	// set命令
	CAutoRedisReply autoR;
	redisReply* r = (redisReply*)redisCommand(m_redisCtx, os.str().c_str());
	if (NULL == r) 
	{
		ERROR("CRedisClient::Lock(),call redisCommand() error,command=%s, redis break connection,m_redisCtx: %p",os.str().c_str(), m_redisCtx);
		m_bConnected = false;
		return false;
	}
	autoR.set(r);

	if (r->type!=REDIS_REPLY_STATUS || r->str==NULL || strcasecmp(r->str, "OK") != 0)
	{
		ERROR("CRedisClient::Lock(),result error, type=%d, command=%s, errmsg=%s", r->type, os.str().c_str(), r->str);
		return false;
	}

	return true;
}

// 解锁
bool CRedisClient::Unlock(RedisLock lockkey)
{
	// 注意:由于hiredis是根据空格来解析cmd参数的,但是lua中肯定有空格,所以这里的命令要分开格式化;不要像上面的那样直接用ostringstream来合成所有的字符串
	char script[256] = {0};
	sprintf(script, "if redis.call('get', KEYS[1]) == '%d' then return redis.call('del', KEYS[1]) else return 0 end", lockkey.randval);

	CAutoRedisReply autoR;
	// 注意命令格式,不要把参数key格式化到script中,那样会报参数个数不够的错误
	redisReply* r = (redisReply*)redisCommand(m_redisCtx, "eval %s 1 %s", script, lockkey.key.c_str());
	if (NULL == r) 
	{
		ERROR("CRedisClient::Lock(),call redisCommand() error,command=%s, redis break connection,m_redisCtx: %p",script, m_redisCtx);
		m_bConnected = false;
		return false;
	}
	autoR.set(r);

	if (r->type!=REDIS_REPLY_INTEGER || r->integer!= 1 || r->str != NULL)
	{
		ERROR("CRedisClient::Lock(),result error, type=%d, command=%s, r->interger=%d, errmsg=%s", r->type, script, r->integer, r->str);
		return false;
	}

	return true;
}

 

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring提供了对Redis分布式锁的支持,可以通过Spring的RedisTemplate或LettuceConnectionFactory来实现。下面是一个简单的实现示例: 1. 首先,在Spring配置文件中配置Redis连接池: ``` <bean id="redisConnectionFactory" class="org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory"> <constructor-arg name="hostName" value="${redis.host}"/> <constructor-arg name="port" value="${redis.port}"/> <constructor-arg name="password" value="${redis.password}"/> </bean> <bean id="redisTemplate" class="org.springframework.data.redis.core.RedisTemplate"> <property name="connectionFactory" ref="redisConnectionFactory"/> <property name="keySerializer"> <bean class="org.springframework.data.redis.serializer.StringRedisSerializer"/> </property> <property name="valueSerializer"> <bean class="org.springframework.data.redis.serializer.StringRedisSerializer"/> </property> </bean> ``` 2. 然后,定义一个Redis分布式锁的工具类: ``` @Component public class RedisLockUtil { private static final long LOCK_EXPIRE_TIME = 30000; // 过期时间,单位毫秒 private static final String LOCK_PREFIX = "lock:"; // 前缀 @Autowired private RedisTemplate<String, String> redisTemplate; public boolean lock(String key) { String lockKey = LOCK_PREFIX + key; long now = System.currentTimeMillis(); long expireTime = now + LOCK_EXPIRE_TIME; Boolean result = redisTemplate.opsForValue().setIfAbsent(lockKey, String.valueOf(expireTime)); if (result != null && result) { redisTemplate.expire(lockKey, LOCK_EXPIRE_TIME, TimeUnit.MILLISECONDS); return true; } String oldExpireTime = redisTemplate.opsForValue().get(lockKey); if (oldExpireTime != null && Long.parseLong(oldExpireTime) < now) { String newExpireTime = redisTemplate.opsForValue().getAndSet(lockKey, String.valueOf(expireTime)); if (newExpireTime != null && newExpireTime.equals(oldExpireTime)) { redisTemplate.expire(lockKey, LOCK_EXPIRE_TIME, TimeUnit.MILLISECONDS); return true; } } return false; } public void unlock(String key) { String lockKey = LOCK_PREFIX + key; redisTemplate.delete(lockKey); } } ``` 该工具类定义了两个方法:lock和unlock。其中,lock方法实现Redis分布式锁的获取,unlock方法实现Redis分布式锁的释放。在lock方法中,首先将的过期时间设置为当前时间+30秒,并通过Redis的setIfAbsent方法尝试获取;如果获取成功,则返回true;否则,通过get方法获取的过期时间,如果已经过期,则通过getAndSet方法更新的过期时间并获取旧的过期时间,然后判断旧的过期时间是否等于获取到的过期时间,如果相等,则说明获取到了,返回true。在unlock方法中,直接通过delete方法删除。 3. 最后,在需要使用分布式锁的地方,注入RedisLockUtil即可使用分布式锁: ``` @Autowired private RedisLockUtil redisLockUtil; public void doSomethingWithLock(String key) { if (redisLockUtil.lock(key)) { try { // 获取成功后执行业务逻辑 // ... } finally { redisLockUtil.unlock(key); } } else { // 获取失败后的处理 // ... } } ``` 上面是一个简单的Spring实现Redis分布式锁的示例,如果您有更多的问题,请继续提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值