zoukankan      html  css  js  c++  java
  • Redis实现分布式锁

    需求

      最近负责的一个项目是微服务开发,我所开发的服务里有个定时任务,如果在集群环境下仅仅使用@scheduled,会发生定时任务幂等的问题。

      可以采取的解决方案:

      1.通过在配置文件设置定时任务开关,只允许一台开启定时任务。考虑到使用了nacos作为配置中心,单节点定时任务对单台服务器的压力以及单点故障问题,不采用。

      2.搭建一套分布式任务调度系统。考虑到时间紧急及目前定时任务比较少,不采用。

      3.使用分布式锁。由于项目本身就使用到redis,而且开发成本低,采用。

    分布式锁介绍

      什么是分布式锁?

      分布式锁是控制分布式系统之间共同访问共享资源的⼀种锁实现

      为什么要用分布式锁?

      在单节点系统中,我们可以通过synchronized和lock保证同⼀个⽅法在同一个时刻被同一个线程调用。但是在分布式系统中,这就不管用了,因此,我们使用分布式锁的目的是保证在分布式部署的应⽤集群中,同⼀个⽅法在同⼀时刻只能被⼀台机器上的⼀个线程执⾏。就像需求中提到到,同一时刻只有一台机子在执行定时任务。

    实现原理和思路

      原理:通过Redis的setnx key value命令,当key不存在时,set一个key为val的字符串,返回1;若key存在,则什么都不做,返回0。

      思路:

      1.获取锁的时候,使用setnx加锁,并使用expire命令为锁添加一个超时时间,避免宕机或网络原因发生导致锁不能释放发生死锁。

      2.锁的value值为当前服务器的ip+端口(考虑在定时任务中使用),大家也可以用UUID,在释放锁的时候进行判断,避免释放到不属于自己的锁。

    代码实现(基于SpringBoot2.0开发)

      第一步:Maven依赖

    1         <dependency>
    2             <groupId>org.springframework.boot</groupId>
    3             <artifactId>spring-boot-starter-data-redis</artifactId>
    4         </dependency>

      第二步:Redis参数配置

      在application.properties中添加redis的链接地址

    1 spring.redis.database = 0
    2 spring.redis.host = 127.0.0.1
    3 spring.redis.port = 6379
    4 spring.redis.password =

      创建RedisConfig

     1 @Configuration
     2 public class RedisConfig {
     3 
     4     @Bean
     5     public RedisTemplate<String, String> redisTemplate(RedisConnectionFactory factory) {
     6         RedisTemplate<String, String> redisTemplate = new RedisTemplate<>();
     7         redisTemplate.setConnectionFactory(factory);
     8         return redisTemplate;
     9     }
    10 
    11     @Bean
    12     public KeyGenerator simpleKeyGenerator() {
    13         return (o, method, objects) -> {
    14             StringBuilder stringBuilder = new StringBuilder();
    15             stringBuilder.append(o.getClass().getSimpleName());
    16             stringBuilder.append(".");
    17             stringBuilder.append(method.getName());
    18             stringBuilder.append("[");
    19             for (Object obj : objects) {
    20                 stringBuilder.append(obj.toString());
    21             }
    22             stringBuilder.append("]");
    23 
    24             return stringBuilder.toString();
    25         };
    26     }
    27 
    28     @Bean
    29     public CacheManager cacheManager(RedisConnectionFactory redisConnectionFactory) {
    30         return new RedisCacheManager(
    31                 RedisCacheWriter.nonLockingRedisCacheWriter(redisConnectionFactory),
    32                 this.getRedisCacheConfigurationWithTtl(600), 
    33                 this.getRedisCacheConfigurationMap() 
    34         );
    35     }
    36 
    37     private Map<String, RedisCacheConfiguration> getRedisCacheConfigurationMap() {
    38         Map<String, RedisCacheConfiguration> redisCacheConfigurationMap = new HashMap<>();
    39         redisCacheConfigurationMap.put("UserInfoList", this.getRedisCacheConfigurationWithTtl(100));
    40         redisCacheConfigurationMap.put("UserInfoListAnother", this.getRedisCacheConfigurationWithTtl(18000));
    41 
    42         return redisCacheConfigurationMap;
    43     }
    44 
    45     private RedisCacheConfiguration getRedisCacheConfigurationWithTtl(Integer seconds) {
    46         Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<>(Object.class);
    47         ObjectMapper om = new ObjectMapper();
    48         om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
    49         om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
    50         jackson2JsonRedisSerializer.setObjectMapper(om);
    51 
    52         RedisCacheConfiguration redisCacheConfiguration = RedisCacheConfiguration.defaultCacheConfig();
    53         redisCacheConfiguration = redisCacheConfiguration.serializeValuesWith(
    54                 RedisSerializationContext
    55                         .SerializationPair
    56                         .fromSerializer(jackson2JsonRedisSerializer)
    57         ).entryTtl(Duration.ofSeconds(seconds));
    58 
    59         return redisCacheConfiguration;
    60     }
    61 
    62 }

      第三步:创建RedisService,实现加锁与释放,加锁的过程不是原子操作,理论上会发生上锁成功但是没有设置过期时间的可能,这里改进的话可以使用Lua脚本来保证原子性操作。

     1 @Service
     2 public class RedisService {
     3 
     4     @Autowired
     5     private RedisTemplate redisTemplate;
     6 
     7 
     8     /**
     9      * 获取分布式锁
    10      *
    11      * @param lockName  锁名称
    12      * @param lockValue 锁值
    13      * @param lockValue 过期时间
    14      * @return
    15      */
    16     public boolean lock(String lockName, String lockValue, Long expireTime) {
    17         boolean result = false;
    18         try {
    19             result = redisTemplate.opsForValue().setIfAbsent(lockName, lockValue);
    20             if (result) {
    21                 redisTemplate.expire(lockName, expireTime, TimeUnit.SECONDS);
    22             }
    23         } catch (Exception e) {
    24             e.printStackTrace();
    25         }
    26         return result;
    27     }
    28 
    29     /**
    30      * 释放分布式锁
    31      *
    32      * @param lockName  锁名称
    33      * @param lockValue 锁值
    34      * @return
    35      */
    36     public boolean release(String lockName, String lockValue) {
    37         boolean result = false;
    38         try {
    39             if (lockValue.equals(redisTemplate.opsForValue().get(lockName))) {
    40                 return redisTemplate.delete(lockName);
    41             }
    42         } catch (Exception e) {
    43             e.printStackTrace();
    44         }
    45         return result;
    46     }
    47 
    48 }

      第四步:代码测试

     1 @Service
     2 @Slf4j
     3 public class MyLockJob {
     4 
     5     private static String LOCK_NAME = "my_lock";
     6 
     7     @Value("${server.port}")
     8     private String serverPort;
     9     @Autowired
    10     private RedisService redisService;
    11 
    12 
    13     @Scheduled(cron = "0/9 * * * * *")
    14     public void lockJob() throws Exception {
    15         boolean lockResult = false;
    16         try {
    17 
    18             String lockValue = InetAddress.getLocalHost().getHostAddress() + ":" + serverPort;
    19             lockResult = redisService.lock(LOCK_NAME, lockValue, 10L);
    20 
    21             // 获取锁成功
    22             if (lockResult) {
    23                 log.info("地址:" + InetAddress.getLocalHost().getHostAddress() + ":" + serverPort + "获取锁成功!");
    24                 Thread.sleep(5000);
    25                 return;
    26             }
    27         } catch (Exception e) {
    28             log.error("lock error", e);
    29         } finally {
    30             // 根据锁的名称及锁的值释放锁
    31             if (lockResult) {
    32                 String releaseValue = InetAddress.getLocalHost().getHostAddress() + ":" + serverPort;
    33                 boolean releaseResult = redisService.release(LOCK_NAME, releaseValue);
    34                 if (releaseResult) {
    35                     log.info(InetAddress.getLocalHost().getHostAddress() + ":" + serverPort + "释放锁成功!");
    36                 }
    37             }
    38         }
    39     }
    40 
    41 }

    结果展示

      本地起了四台服务8080~8083,模拟多台服务器争抢锁及释放锁的过程,这里展示一分钟内获取锁和释放锁的情况。

      8080端口获取锁:

      8081端口获取锁:

      8082端口有点倒霉,没拿到锁。

      8083端口获取锁:

       可以看到,同一时刻,只有一个服务能拿到锁,并且获取锁及释放锁为同一服务,需求完成。

  • 相关阅读:
    来看看面试必问的HashMap,一次彻底帮你搞定HashMap源码
    深入浅出!springboot从入门到精通,实战开发全套教程!
    讲一讲Java的字符串常量池,看完你的思路就清晰了
    面向对象的这些核心技术,你掌握后包你面试无忧
    他凭借这70份PDF,3170页文件,成功斩获了含BATJ所有的offer
    springboot实战开发全套教程,让开发像搭积木一样简单!Github星标已上10W+!
    这行代码告诉你!为什么你地下城与勇士(DNF)的装备强化老是失败?
    精益求精!Spring Boot 知识点全面回顾,带你重新细读源码!
    太妙了!Spring boot 整合 Mybatis Druid,还能配置监控?
    putchar(".:-=+*#%@"[(int)(d * 5.0f)])
  • 原文地址:https://www.cnblogs.com/maguanyue/p/11742438.html
Copyright © 2011-2022 走看看