分布式锁Redisson 防止用户重复点击按钮实现!

一:什么事Redisson?

官网:https://redisson.org/

百度百科:
    Redisson是架设在Redis基础上的一个java驻内存数据网格,(In-Memory Data Grid)。【Redis官方推荐】
    Redisson是基于NIO和Netty框架上,充分的利用Redis键值数据库提供的一系列优势,
    在java实用工具包中,常用接口的基础上,为使用者提供了一些列的具有分布式特性的
    工具类,使得原本作为协调单机多线程并发程序的工具包获得了协调分布式多机多线程
    并发系统的能力,大大降低了设计和研发大规模分布式系统难度

二:本篇文章旨在介绍Redision简单的应用(防止用户重复点击按钮实现)

1.Maven项目引入

<!-- JDK 1.8+ compatible -->
<dependency>
   <groupId>org.redisson</groupId>
   <artifactId>redisson</artifactId>
   <version>3.5.4</version>
</dependency>  
 
<!-- JDK 1.6+ compatible -->
<dependency>
   <groupId>org.redisson</groupId>
   <artifactId>redisson</artifactId>
   <version>2.10.4</version>
</dependency>

1.1 用于连接Redis服务器 创建不同的Redis Server对应的客户端对象工具类

package com.rongdu.cashloan.core.redisson;

import org.apache.commons.lang3.StringUtils;
import org.redisson.Redisson;
import org.redisson.api.*;
import org.redisson.config.Config;
import org.redisson.config.SingleServerConfig;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/***
 * Redis client的辅助工具类
 * 用于连接Redis服务器 创建不同的Redis Server对应的客户端对象
 * @author wangnian
 * 博客地址:http://my.oschina.net/wangnian
 */
public class RedisUtils {

   private static  Logger logger= LoggerFactory.getLogger(RedisUtils.class);

   private static RedisUtils redisUtils;

   private static RedissonClient redissonClient;

   private RedisUtils(){}

   /**
    * 提供单例模式
    * @return
    */
   public static RedisUtils getInstance(){
      if(redisUtils==null) {
         synchronized (RedisUtils.class) {
            if(redisUtils==null){
               redisUtils=new RedisUtils();
            }
         }
      }
      return redisUtils;
   }


   /**
    * 使用config创建Redisson
    * Redisson是用于连接Redis Server的基础类
    * @param config
    * @return
    */
   public RedissonClient getRedisson(Config config){
      if(redissonClient==null){
         synchronized (RedisUtils.class) {
            if(redissonClient==null){
               redissonClient=Redisson.create(config);
            }
         }
      }
      return redissonClient;
   }

   /**
    * 使用ip地址和端口创建Redisson
    * @param ip
    * @param port
    * @return
    */
   public RedissonClient getRedisson(String ip,String port,String passwd){

      if(!ip.startsWith("http://")){
         ip = "http://"+ip;
      }

      logger.info("RedisUtils-getRedisson-传入参数-ip:"+ip+"-port:"+port+"-passwd:"+passwd);
      Config config=new Config();

      SingleServerConfig singleServerConfig = config.useSingleServer();

      singleServerConfig.setAddress(ip+":"+port).
              setTimeout(200000).
              setRetryAttempts(10).
              // .setIdleConnectionTimeout(10000)
                      setRetryInterval(2000).
              //.setFailedAttempts(20)
                      setSubscriptionsPerConnection(5).
              setSubscriptionConnectionPoolSize(10).
              setConnectionPoolSize(64).
              setDatabase(0);

      if(StringUtils.isNotBlank(passwd)){
         singleServerConfig.setPassword(passwd);
      }

      if(redissonClient==null){
         synchronized (RedisUtils.class) {
            if(redissonClient==null){
               redissonClient=Redisson.create(config);
            }
         }
      }
      return redissonClient;
   }

   /**
    * 关闭Redisson客户端连接
    * @param redisson
    */
   public void closeRedisson(RedissonClient redisson){
      redisson.shutdown();
   }

   /**
    * 获取字符串对象
    * @param redisson
    * @param objectName
    * @return
    */
   public <T> RBucket<T> getRBucket(RedissonClient redisson, String objectName){
      RBucket<T> bucket=redisson.getBucket(objectName);
      return bucket;
   }

   /**
    * 获取Map对象
    * @param redisson
    * @param objectName
    * @return
    */
   public <K,V> RMap<K, V> getRMap(RedissonClient redisson, String objectName){
      RMap<K, V> map=redisson.getMap(objectName);
      return map;
   }

   /**
    * 获取有序集合
    * @param redisson
    * @param objectName
    * @return
    */
   public <V> RSortedSet<V> getRSortedSet(RedissonClient redisson, String objectName){
      RSortedSet<V> sortedSet=redisson.getSortedSet(objectName);
      return sortedSet;
   }

   /**
    * 获取集合
    * @param redisson
    * @param objectName
    * @return
    */
   public <V> RSet<V> getRSet(RedissonClient redisson, String objectName){
      RSet<V> rSet=redisson.getSet(objectName);
      return rSet;
   }

   /**
    * 获取列表
    * @param redisson
    * @param objectName
    * @return
    */
   public <V> RList<V> getRList(RedissonClient redisson,String objectName){
      RList<V> rList=redisson.getList(objectName);
      return rList;
   }

   /**
    * 获取队列
    * @param redisson
    * @param objectName
    * @return
    */
   public <V> RQueue<V> getRQueue(RedissonClient redisson,String objectName){
      RQueue<V> rQueue=redisson.getQueue(objectName);
      return rQueue;
   }

   /**
    * 获取双端队列
    * @param redisson
    * @param objectName
    * @return
    */
   public <V> RDeque<V> getRDeque(RedissonClient redisson,String objectName){
      RDeque<V> rDeque=redisson.getDeque(objectName);
      return rDeque;
   }

   /**
    * 此方法不可用在Redisson 1.2 中 
    * 在1.2.2版本中 可用
    * @param redisson
    * @param objectName
    * @return
    */
   /**
    public <V> RBlockingQueue<V> getRBlockingQueue(RedissonClient redisson,String objectName){
    RBlockingQueue rb=redisson.getBlockingQueue(objectName);
    return rb;
    }*/

   /**
    * 获取锁
    * @param redisson
    * @param objectName
    * @return
    */
   public RLock getRLock(RedissonClient redisson,String objectName){
      RLock rLock=redisson.getLock(objectName);
      return rLock;
   }

   /**
    * 获取原子数
    * @param redisson
    * @param objectName
    * @return
    */
   public RAtomicLong getRAtomicLong(RedissonClient redisson,String objectName){
      RAtomicLong rAtomicLong=redisson.getAtomicLong(objectName);
      return rAtomicLong;
   }

   /**
    * 获取记数锁
    * @param redisson
    * @param objectName
    * @return
    */
   public RCountDownLatch getRCountDownLatch(RedissonClient redisson,String objectName){
      RCountDownLatch rCountDownLatch=redisson.getCountDownLatch(objectName);
      return rCountDownLatch;
   }

   /**
    * 获取消息的Topic
    * @param redisson
    * @param objectName
    * @return
    */
   public <M> RTopic<M> getRTopic(RedissonClient redisson,String objectName){
      RTopic<M> rTopic=redisson.getTopic(objectName);
      return rTopic;
   }


}

1.2:防止用户频繁请求代码实现

public void commitUserReq(HttpServletRequest request){
        final RedissonClient redisson = RedisUtils.getInstance().getRedisson(redisIp,redisPort,redisPasswd);
        // 防重复提交锁
        RLock lock = redisson.getLock(orderId + "COMMIT_ORDER" + userId);
        try {
            boolean res = lock.tryLock(0, 10, TimeUnit.SECONDS);
            if (!res){
                throw new SimpleMessageException("操作太频繁!");
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        // 防止正在请求过程再发起请求
        boolean isBack = false;
        RLock lock2 = redisson.getLock(orderId + "LOCK_BUTTON" + userId);
        lock2.lock(10,TimeUnit.SECONDS);
        if (redisClient.exists("BUTTON_PAY_"+ userId)){
            isBack = true;
        }else{
            redisClient.setObject("BUTTON_PAY_"+ userId,borrowId,20*60);
        }
        lock2.unlock();
        if (isBack){
            logger.info("系统提交已经开始");
            throw new SimpleMessageException("你有请求正在处理,请20分钟后再试!");
        }
		System.out.println("处理业务代码==========正常执行");
		
		}

 

 

 

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值