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

一:什么事Redisson?

官网:https://redisson.org/

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

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

1.Maven项目引入



   org.redisson
   redisson
   3.5.4
  
 


   org.redisson
   redisson
   2.10.4

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  RBucket getRBucket(RedissonClient redisson, String objectName){
      RBucket bucket=redisson.getBucket(objectName);
      return bucket;
   }

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

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

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

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

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

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

   /**
    * 此方法不可用在Redisson 1.2 中 
    * 在1.2.2版本中 可用
    * @param redisson
    * @param objectName
    * @return
    */
   /**
    public  RBlockingQueue 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  RTopic getRTopic(RedissonClient redisson,String objectName){
      RTopic 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("处理业务代码==========正常执行");
		
		}

 

 

 

你可能感兴趣的:(Redis)