实现分布式锁的几种方案

1.实现分布式锁的几种方案

    1.Redis实现   (推荐)

    2.Zookeeper实现

    3.数据库实现

Redis实现分布式锁
*
* 在集群等多服务器中经常使用到同步处理一下业务,这是普通的事务是满足不了业务需求,需要分布式锁
*
* 分布式锁的常用3种实现:
*               0.数据库乐观锁实现
*               1.Redis实现   ---  使用redis的setnx()、get()、getset()方法,用于分布式锁,解决死锁问题
*               2.Zookeeper实现
*                      参考:http://surlymo.iteye.com/blog/2082684
*                            http://www.jb51.net/article/103617.htm
*                            http://www.hollischuang.com/archives/1716?utm_source=tuicool&utm_medium=referral
*                   1、实现原理:
基于zookeeper瞬时有序节点实现的分布式锁,其主要逻辑如下(该图来自于IBM网站)。大致思想即为:每个客户端对某个功能加锁时,在zookeeper上的与该功能对应的指定节点的目录下,生成一个唯一的瞬时有序节点。判断是否获取锁的方式很简单,只需要判断有序节点中序号最小的一个。当释放锁的时候,只需将这个瞬时节点删除即可。同时,其可以避免服务宕机导致的锁无法释放,而产生的死锁问题。
2、优点
锁安全性高,zk可持久化
3、缺点
性能开销比较高。因为其需要动态产生、销毁瞬时节点来实现锁功能。
4、实现
可以直接采用zookeeper第三方库curator即可方便地实现分布式锁
*
* Redis实现分布式锁的原理:
*   1.通过setnx(lock_timeout)实现,如果设置了锁返回1, 已经有值没有设置成功返回0
*   2.死锁问题:通过实践来判断是否过期,如果已经过期,获取到过期时间get(lockKey),然后getset(lock_timeout)判断是否和get相同,
*     相同则证明已经加锁成功,因为可能导致多线程同时执行getset(lock_timeout)方法,这可能导致多线程都只需getset后,对于判断加锁成功的线程,
*     再加expire(lockKey, LOCK_TIMEOUT, TimeUnit.MILLISECONDS)过期时间,防止多个线程同时叠加时间,导致锁时效时间翻倍
*   3.针对集群服务器时间不一致问题,可以调用redis的time()获取当前时间

2.Redis分分布式锁的代码实现

   1.定义锁接口

[java] view plain copy

  1. package com.jay.service.redis;  
  2.   
  3. /** 
  4.  * Redis分布式锁接口 
  5.  * Created by hetiewei on 2017/4/7. 
  6.  */  
  7. public interface RedisDistributionLock {  
  8.     /** 
  9.      * 加锁成功,返回加锁时间 
  10.      * @param lockKey 
  11.      * @param threadName 
  12.      * @return 
  13.      */  
  14.     public long lock(String lockKey, String threadName);  
  15.   
  16.     /** 
  17.      * 解锁, 需要更新加锁时间,判断是否有权限 
  18.      * @param lockKey 
  19.      * @param lockValue 
  20.      * @param threadName 
  21.      */  
  22.     public void unlock(String lockKey, long lockValue, String threadName);  
  23.   
  24.     /** 
  25.      * 多服务器集群,使用下面的方法,代替System.currentTimeMillis(),获取redis时间,避免多服务的时间不一致问题!!! 
  26.      * @return 
  27.      */  
  28.     public long currtTimeForRedis();  
  29. }  

   2.定义锁实现

[java] view plain copy

  1. package com.jay.service.redis.impl;  
  2.   
  3. import com.jay.service.redis.RedisDistributionLock;  
  4. import org.slf4j.Logger;  
  5. import org.slf4j.LoggerFactory;  
  6. import org.springframework.dao.DataAccessException;  
  7. import org.springframework.data.redis.connection.RedisConnection;  
  8. import org.springframework.data.redis.core.RedisCallback;  
  9. import org.springframework.data.redis.core.StringRedisTemplate;  
  10. import org.springframework.data.redis.serializer.RedisSerializer;  
  11.   
  12. import java.util.concurrent.TimeUnit;  
  13.   
  14. /** 
  15.  * Created by hetiewei on 2017/4/7. 
  16.  */  
  17. public class RedisLockImpl implements RedisDistributionLock {  
  18.   
  19.     //加锁超时时间,单位毫秒, 即:加锁时间内执行完操作,如果未完成会有并发现象  
  20.     private static final long LOCK_TIMEOUT = 5*1000;  
  21.   
  22.     private static final Logger LOG = LoggerFactory.getLogger(RedisLockImpl.class);  
  23.   
  24.     private StringRedisTemplate redisTemplate;  
  25.   
  26.     public RedisLockImpl(StringRedisTemplate redisTemplate) {  
  27.         this.redisTemplate = redisTemplate;  
  28.     }  
  29.   
  30.     /** 
  31.      * 加锁 
  32.      * 取到锁加锁,取不到锁一直等待知道获得锁 
  33.      * @param lockKey 
  34.      * @param threadName 
  35.      * @return 
  36.      */  
  37.     @Override  
  38.     public synchronized long lock(String lockKey, String threadName) {  
  39.         LOG.info(threadName+"开始执行加锁");  
  40.         while (true){ //循环获取锁  
  41.             //锁时间  
  42.             Long lock_timeout = currtTimeForRedis()+ LOCK_TIMEOUT +1;  
  43.             if (redisTemplate.execute(new RedisCallback<Boolean>() {  
  44.                 @Override  
  45.                 public Boolean doInRedis(RedisConnection redisConnection) throws DataAccessException {  
  46.                     //定义序列化方式  
  47.                     RedisSerializer<String> serializer = redisTemplate.getStringSerializer();  
  48.                     byte[] value = serializer.serialize(lock_timeout.toString());  
  49.                     boolean flag = redisConnection.setNX(lockKey.getBytes(), value);  
  50.                     return flag;  
  51.                 }  
  52.             })){  
  53.                 //如果加锁成功  
  54.                 LOG.info(threadName +"加锁成功 ++++ 111111");  
  55.                 //设置超时时间,释放内存  
  56.                 redisTemplate.expire(lockKey, LOCK_TIMEOUT, TimeUnit.MILLISECONDS);  
  57.                 return lock_timeout;  
  58.             }else {  
  59.                 //获取redis里面的时间  
  60.                 String result = redisTemplate.opsForValue().get(lockKey);  
  61.                 Long currt_lock_timeout_str = result==null?null:Long.parseLong(result);  
  62.                 //锁已经失效  
  63.                 if (currt_lock_timeout_str != null && currt_lock_timeout_str < System.currentTimeMillis()){  
  64.                     //判断是否为空,不为空时,说明已经失效,如果被其他线程设置了值,则第二个条件判断无法执行  
  65.                     //获取上一个锁到期时间,并设置现在的锁到期时间  
  66.                     Long old_lock_timeout_Str = Long.valueOf(redisTemplate.opsForValue().getAndSet(lockKey, lock_timeout.toString()));  
  67.                     if (old_lock_timeout_Str != null && old_lock_timeout_Str.equals(currt_lock_timeout_str)){  
  68.                         //多线程运行时,多个线程签好都到了这里,但只有一个线程的设置值和当前值相同,它才有权利获取锁  
  69.                         LOG.info(threadName + "加锁成功 ++++ 22222");  
  70.                         //设置超时间,释放内存  
  71.                         redisTemplate.expire(lockKey, LOCK_TIMEOUT, TimeUnit.MILLISECONDS);  
  72.   
  73.                         //返回加锁时间  
  74.                         return lock_timeout;  
  75.                     }  
  76.                 }  
  77.             }  
  78.   
  79.             try {  
  80.                 LOG.info(threadName +"等待加锁, 睡眠100毫秒");  
  81. //                TimeUnit.MILLISECONDS.sleep(100);  
  82.                 TimeUnit.MILLISECONDS.sleep(200);  
  83.             } catch (InterruptedException e) {  
  84.                 e.printStackTrace();  
  85.             }  
  86.         }  
  87.     }  
  88.   
  89.     /** 
  90.      * 解锁 
  91.      * @param lockKey 
  92.      * @param lockValue 
  93.      * @param threadName 
  94.      */  
  95.     @Override  
  96.     public synchronized void unlock(String lockKey, long lockValue, String threadName) {  
  97.         LOG.info(threadName + "执行解锁==========");//正常直接删除 如果异常关闭判断加锁会判断过期时间  
  98.         //获取redis中设置的时间  
  99.         String result = redisTemplate.opsForValue().get(lockKey);  
  100.         Long currt_lock_timeout_str = result ==null?null:Long.valueOf(result);  
  101.   
  102.         //如果是加锁者,则删除锁, 如果不是,则等待自动过期,重新竞争加锁  
  103.         if (currt_lock_timeout_str !=null && currt_lock_timeout_str == lockValue){  
  104.             redisTemplate.delete(lockKey);  
  105.             LOG.info(threadName + "解锁成功------------------");  
  106.         }  
  107.     }  
  108.   
  109.     /** 
  110.      * 多服务器集群,使用下面的方法,代替System.currentTimeMillis(),获取redis时间,避免多服务的时间不一致问题!!! 
  111.      * @return 
  112.      */  
  113.     @Override  
  114.     public long currtTimeForRedis(){  
  115.         return redisTemplate.execute(new RedisCallback<Long>() {  
  116.             @Override  
  117.             public Long doInRedis(RedisConnection redisConnection) throws DataAccessException {  
  118.                 return redisConnection.time();  
  119.             }  
  120.         });  
  121.     }  
  122.   
  123. }  

  3.分布式锁验证

      

[java] view plain copy

  1. @RestController  
  2. @RequestMapping("/distribution/redis")  
  3. public class RedisLockController {  
  4.   
  5.     private static final String LOCK_NO = "redis_distribution_lock_no_";  
  6.   
  7.     private static int i = 0;  
  8.   
  9.     private ExecutorService service;  
  10.   
  11.     @Autowired  
  12.     private StringRedisTemplate redisTemplate;  
  13.   
  14.     /** 
  15.      * 模拟1000个线程同时执行业务,修改资源 
  16.      * 
  17.      * 使用线程池定义了20个线程 
  18.      * 
  19.      */  
  20.     @GetMapping("lock1")  
  21.     public void testRedisDistributionLock1(){  
  22.   
  23.         service = Executors.newFixedThreadPool(20);  
  24.   
  25.         for (int i=0;i<1000;i++){  
  26.             service.execute(new Runnable() {  
  27.                 @Override  
  28.                 public void run() {  
  29.                     task(Thread.currentThread().getName());  
  30.                 }  
  31.             });  
  32.         }  
  33.   
  34.     }  
  35.   
  36.     @GetMapping("/{key}")  
  37.     public String getValue(@PathVariable("key") String key){  
  38.         Serializable result = redisTemplate.opsForValue().get(key);  
  39.         return result.toString();  
  40.     }  
  41.   
  42.     private void task(String name) {  
  43. //        System.out.println(name + "任务执行中"+(i++));  
  44.   
  45.         //创建一个redis分布式锁  
  46.         RedisLockImpl redisLock = new RedisLockImpl(redisTemplate);  
  47.         //加锁时间  
  48.         Long lockTime;  
  49.         if ((lockTime = redisLock.lock((LOCK_NO+1)+"", name))!=null){  
  50.             //开始执行任务  
  51.             System.out.println(name + "任务执行中"+(i++));  
  52.             //任务执行完毕 关闭锁  
  53.             redisLock.unlock((LOCK_NO+1)+"", lockTime, name);  
  54.         }  
  55.   
  56.     }  
  57. }  

4.结果验证:

      在Controller中模拟了1000个线程,通过线程池方式提交,每次20个线程抢占分布式锁,抢到分布式锁的执行代码,没抢到的等待

     结果如下:

   

2017-04-07 16:27:17.385  INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-4等待加锁, 睡眠100毫秒
2017-04-07 16:27:17.385  INFO 8652 --- [pool-2-thread-7] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-7解锁成功------------------
        2017-04-07 16:27:17.391  INFO 8652 --- [pool-2-thread-5] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-5加锁成功 ++++ 111111
pool-2-thread-5任务执行中994
2017-04-07 16:27:17.391  INFO 8652 --- [pool-2-thread-5] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-5执行解锁==========
        2017-04-07 16:27:17.391  INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-1等待加锁, 睡眠100毫秒
2017-04-07 16:27:17.391  INFO 8652 --- [pool-2-thread-5] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-5解锁成功------------------
        2017-04-07 16:27:17.397  INFO 8652 --- [pool-2-thread-6] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-6加锁成功 ++++ 111111
pool-2-thread-6任务执行中995
2017-04-07 16:27:17.398  INFO 8652 --- [pool-2-thread-6] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-6执行解锁==========
        2017-04-07 16:27:17.398  INFO 8652 --- [pool-2-thread-6] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-6解锁成功------------------
        2017-04-07 16:27:17.400  INFO 8652 --- [ool-2-thread-19] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-19加锁成功 ++++ 111111
pool-2-thread-19任务执行中996
2017-04-07 16:27:17.400  INFO 8652 --- [ool-2-thread-19] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-19执行解锁==========
        2017-04-07 16:27:17.400  INFO 8652 --- [ool-2-thread-19] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-19解锁成功------------------
        2017-04-07 16:27:17.571  INFO 8652 --- [ool-2-thread-11] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-11加锁成功 ++++ 111111
pool-2-thread-11任务执行中997
2017-04-07 16:27:17.572  INFO 8652 --- [ool-2-thread-11] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-11执行解锁==========
        2017-04-07 16:27:17.572  INFO 8652 --- [ool-2-thread-11] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-11解锁成功------------------
        2017-04-07 16:27:17.585  INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-4加锁成功 ++++ 111111
pool-2-thread-4任务执行中998
2017-04-07 16:27:17.586  INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-4执行解锁==========
        2017-04-07 16:27:17.586  INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-4解锁成功------------------
        2017-04-07 16:27:17.591  INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-1加锁成功 ++++ 111111
pool-2-thread-1任务执行中999
2017-04-07 16:27:17.591  INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-1执行解锁==========
        2017-04-07 16:27:17.591  INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl   : pool-2-thread-1解锁成功

猜你喜欢

转载自blog.csdn.net/u014352080/article/details/79925237