Laravel 实现redis分布式锁

5 篇文章 0 订阅

Redis几种架构

Redis发展到现在,几种常见的部署架构有:

单机模式;
主从模式;
哨兵模式;
集群模式;

从分布式锁的角度来说, 无论是单机模式、主从模式、哨兵模式、集群模式,其原理都是类同的。 只是主从模式、哨兵模式、集群模式的更加的高可用、或者更加高并发。 

所以,接下来先基于单机模式,基于Jedis手工造轮子实现自己的分布式锁。

首先看两个命令:

Redis分布式锁机制,主要借助setnx和expire两个命令完成。

setnx命令:

SETNX 是SET if Not eXists的简写。将 key 的值设为 value,当且仅当 key 不存在; 若给定的 key 已经存在,则 SETNX 不做任何动作。

下面为客户端使用示例:

127.0.0.1:6379> set lock "unlock"
OK
127.0.0.1:6379> setnx lock "unlock"
(integer) 0
127.0.0.1:6379> setnx lock "lock"
(integer) 0
127.0.0.1:6379> 

expire命令:

expire命令为 key 设置生存时间,当 key 过期时(生存时间为 0 ),它会被自动删除. 其格式为:

EXPIRE key seconds

下面为客户端使用示例:

127.0.0.1:6379> expire lock 10
(integer) 1
127.0.0.1:6379> ttl lock
8

        $lock = Redis::setnx('cqny',5,1);
        if(!$lock) return $this->apiError('操作太频繁了');
        Redis::expire('cqny',10);

        $block = Cache::lock("GoodLock",10); //返回一个redis对象
        if ($block){
            // Todo 业务逻辑
            sleep(10);
            echo "业务逻辑<br>";
            $block->release(); // 释放锁
            return "秒杀成功";
        }
        return 12;

 

        $suo = Redis::set('123',123,"nx","ex",5);
        if ($suo) Redis::del('123');

 

  • 1
    点赞
  • 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、付费专栏及课程。

余额充值