使用Redisson实现分布式锁最佳实践

废话不多说,直接上代码,企业级拿来即用。
分布式锁组件共5个类,代码结构:

说明:

  1. RedissonConnector: 分布式锁客户端配置类。
  2. RedissonLocker: 分布式锁的工作类。
  3. RedissonLockWorker: 获取锁后的处理类。
  4. RedissonLockService: 分布式锁的接口类。
  5. RedissonLockException: 分布式锁异常类。

各个类详细内容:

  1. package com.**.lock;
    
    import org.redisson.Redisson;
    import org.redisson.api.RedissonClient;
    import org.redisson.config.Config;
    import org.springframework.beans.factory.annotation.Value;
    import org.springframework.stereotype.Component;
    
    import javax.annotation.PostConstruct;
    
    /**
     * @author yuan
     * redisson client连接类
     */
    @Component
    public class RedissonConnector {
    
        @Value("${redis.host}")
        private String host;
        @Value("${redis.port}")
        private String port;
        @Value("${redis.password}")
        private String password;
    
        private RedissonClient redissonClient;
    
        @PostConstruct
        public void init() {
            Config config = new Config();
            config.useSingleServer().setAddress(host + ":" + port).setPassword(password);
            redissonClient = Redisson.create(config);
        }
    
        RedissonClient getRedissonClient() {
            return redissonClient;
        }
    
    }
    

     

  2. package com.**.lock;
    
    import com.msunsoft.utils.RedistUtil;
    import org.redisson.api.RLock;
    import org.redisson.api.RedissonClient;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Component;
    
    import java.util.concurrent.TimeUnit;
    
    /**
     * @author yuan
     * redisson锁工作类
     */
    @Component
    public class RedissonLocker implements RedissonLockService {
    
        private final RedissonConnector connector;
    
        @Autowired
        public RedissonLocker(RedissonConnector connector) {
            this.connector = connector;
        }
    
        @Override
        public boolean lock(String sourceName) throws Exception {
            RLock rLock = this.getLock(RedistUtil.lock_prefix + sourceName);
            //尝试加锁,最多等待60秒,加锁lockTime以后自动解锁
            return rLock.tryLock(60, RedistUtil.lock_expire, TimeUnit.SECONDS);
        }
    
        @Override
        public <T> T lock(String sourceName, RedissonLockWorker<T> worker) throws Exception {
            return lock(sourceName, worker, RedistUtil.lock_expire);
        }
    
        @Override
        public <T> T lock(String sourceName, RedissonLockWorker<T> worker, int lockTime) throws Exception {
            RLock rLock = this.getLock(RedistUtil.lock_prefix + sourceName);
            //尝试加锁,最多等待60秒,加锁lockTime以后自动解锁
            boolean success = rLock.tryLock(60, lockTime, TimeUnit.SECONDS);
            if (success) {
                try {
                    return worker.afterLockSuccess();
                } finally {
                    rLock.unlock();
                }
            }
            throw new RedissonLockException();
        }
    
        @Override
        public void unLock(String sourceName) {
            RLock rLock = this.getLock(RedistUtil.lock_prefix + sourceName);
            rLock.unlock();
        }
    
        /**
         * 获取锁
         */
        private RLock getLock(String name) {
            RedissonClient client = connector.getRedissonClient();
            return client.getLock(name);
        }
    }
    

     

  3. package com.**.lock;
    
    /**
     * @author yuan
     * 获取锁后处理类
     */
    public interface RedissonLockWorker<T> {
    
        /**
         * 获取锁成功后执行业务逻辑
         * @return 业务逻辑对象
         * @throws Exception 异常信息
         */
        T afterLockSuccess() throws Exception;
    
    
    }
    

     

  4. package com.**.lock;
    
    /**
     * @author yuan
     * 锁管理类
     */
    public interface RedissonLockService {
    
        /**
         * 加锁
         * @param sourceName 资源名称
         * @return 处理完业务逻辑后返回的数据
         * @throws Exception 异常
         */
        boolean lock(String sourceName) throws Exception;
    
        /**
         * 加锁
         * @param sourceName 资源名称
         * @param worker 获取锁后工作类
         * @return 处理完业务逻辑后返回的数据
         * @throws Exception 异常
         */
        <T> T lock(String sourceName, RedissonLockWorker<T> worker) throws Exception;
    
        /**
         * 加锁
         * @param sourceName 资源名称
         * @param worker 获取锁后工作类
         * @param lockTime 锁超时时间
         * @return 处理完业务逻辑后返回的数据
         * @throws Exception 异常
         */
        <T> T lock(String sourceName, RedissonLockWorker<T> worker, int lockTime) throws Throwable;
    
        /**
         * 锁释放
         * @param sourceName 资源名称
         */
        void unLock(String sourceName);
    }
    

     

  5. package com.**.lock;
    
    /**
     * @author yuan
     * 锁异常类
     */
    public class RedissonLockException extends RuntimeException {
        public RedissonLockException() {
        }
    
        public RedissonLockException(String message) {
            super(message);
        }
    }
    

     

  6. @Autowired
    private RedissonLockService redissonLockService;
    
    @RequestMapping("/wechat/submitEvalInfo")
    public Result submitEvalInfo(@RequestBody List<QuestionnaireEvalDTO> evalList) throws Exception {
            TokenUser user = ThreadContext.get();
            try {
                //sourceName随意,只要保证一定时间内唯一就行
                return redissonLockService.lock(user.getLoginName() + "submitEvalInfo",
                        () -> questionnaireService.submitEvalInfo(evalList, user));
            } catch (UnsupportedEncodingException e) {
                return Result.fail("数据解码失败,请联系运营平台开发人员解决");
            }
        }

     

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 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
发出的红包

打赏作者

程序员·猿先生

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值