欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Mysql、Redis、Redisson分布式锁

程序员文章站 2022-05-04 08:56:59
Mysql代码ProductEntity selectLock(@Param("title") String title);@RequestMapping("mysql/singleLock")...

代码

Mysql

代码

ProductEntity selectLock(@Param("title") String title);
<select id="selectLock" resultType="com.csea.lock.pojo.ProductEntity">
    select * from product where title = #{title} for update
</select>
	@RequestMapping("mysql/singleLock")
    @Transactional(rollbackFor = Exception.class)
    public String mysqlSingleLock() throws Exception {
        log.info("进入了方法");
        ProductEntity selectLock = productMapper.selectLock("苹果");
        if (StringUtils.isEmpty(selectLock)) {
            throw new Exception("没有分布式锁");
        }
        log.info("拿到了锁");

        Thread.sleep(5000);

        return "完成了";
    }

效果

启动8080,8081两个端口进行访问
5秒之后拿到了锁
Mysql、Redis、Redisson分布式锁
Mysql、Redis、Redisson分布式锁

Redis

原理

SET resource_name my_random_value NX PX 30000
resource_name:资源名称
my_random_value:随机值,每个线程随机值不同,用于释放锁时的校验
NX:key不存在时设置成功,反之不超过
PX:过期时间,出现异常情况,锁可以过期失效
利用NX的原子性,多个线程并发时候,只有一个线程可以设置成功;设置成功即可获得锁,继续执行后面的业务;出现异常,锁了有效期,锁就自动释放。

代码

	private final RedisTemplate redisTemplate;

    @GetMapping("/redis/lock")
    public String redisLock() {
        log.info("进入了方法");
        String key = "Csea";
        String value = UUID.randomUUID().toString();

        RedisCallback<Boolean> redisCallback = redisConnection -> {
            // 设置NX
            RedisStringCommands.SetOption setOption = RedisStringCommands.SetOption.ifAbsent();
            // 设置过期时间
            Expiration expiration = Expiration.seconds(30);
            // 序列化Key
            byte[] redisKey = redisTemplate.getKeySerializer().serialize(key);
            // 序列化Value
            byte[] redisValue = redisTemplate.getValueSerializer().serialize(value);
            // 返回 setnx操作结果
            return redisConnection.set(redisKey, redisValue, expiration, setOption);
        };

        // 获取分布式锁
        Boolean lock = (Boolean) redisTemplate.execute(redisCallback);

        if (lock) {
            log.info("拿到了锁");
            try {
                Thread.sleep(5000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                // 释放锁

                // LUA脚本
                String script = "if redis.call(\"get\", KEYS[1]) == ARGV[1] then\n" +
                        "  return redis.call(\"del\",KEYS[1])\n" +
                        "else\n" +
                        "  return 0\n" +
                        "end";
                RedisScript<Boolean> redisScript = RedisScript.of(script, Boolean.class);
                List<String> keyList = Arrays.asList(key);

                Boolean result = (Boolean) redisTemplate.execute(redisScript, keyList, value);
                log.info("释放锁结果:{}", result);
            }
        }
        log.info("执行结束");
        return "执行结束";
    }

效果

启动8080,8081两个端口进行访问
8080没有拿到锁,就直接运行结束了。
Mysql、Redis、Redisson分布式锁
Mysql、Redis、Redisson分布式锁

分布式下的定时任务

redis锁的代码抽离

@Slf4j
public class RedisLock implements AutoCloseable {

    private RedisTemplate redisTemplate;
    private String key;
    private String value;
    private int expireTime;

    public RedisLock(RedisTemplate redisTemplate, String key, int expireTime) {
        this.redisTemplate = redisTemplate;
        this.key = key;
        this.value = UUID.randomUUID().toString();
        this.expireTime = expireTime;
    }

    public boolean getLock() {
        RedisCallback<Boolean> redisCallback = redisConnection -> {
            // 设置NX
            RedisStringCommands.SetOption setOption = RedisStringCommands.SetOption.ifAbsent();
            // 设置过期时间
            Expiration expiration = Expiration.seconds(expireTime);
            // 序列化Key
            byte[] redisKey = redisTemplate.getKeySerializer().serialize(key);
            // 序列化Value
            byte[] redisValue = redisTemplate.getValueSerializer().serialize(value);
            // 返回 setnx操作结果
            return redisConnection.set(redisKey, redisValue, expiration, setOption);
        };

        // 获取分布式锁
        Boolean lock = (Boolean) redisTemplate.execute(redisCallback);
        return lock;
    }

    public boolean unlock() {
        // LUA脚本
        String script = "if redis.call(\"get\", KEYS[1]) == ARGV[1] then\n" +
                "  return redis.call(\"del\",KEYS[1])\n" +
                "else\n" +
                "  return 0\n" +
                "end";
        RedisScript<Boolean> redisScript = RedisScript.of(script, Boolean.class);
        List<String> keyList = Arrays.asList(key);

        Boolean result = (Boolean) redisTemplate.execute(redisScript, keyList, value);
        log.info("释放锁的结果:{}", result);
        return result;
    }

    @Override
    public void close() throws Exception {
        unlock();
    }
}

定时任务代码

@Service
@Slf4j
@RequiredArgsConstructor
public class SchedulerService {

    private final RedisTemplate redisTemplate;


    @Scheduled(cron = "0/5 * * * * ?")
    public void logInfo() {
        try (RedisLock lock = new RedisLock(redisTemplate, "Csea", 30)) {
            if (lock.getLock()) {
                log.info("打印了*************Csea*************");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

效果

启动8080,8081两个端口
Mysql、Redis、Redisson分布式锁
Mysql、Redis、Redisson分布式锁

Redisson

官网
文档
整合SpringBoot

<dependency>
	<groupId>org.redisson</groupId>
	<artifactId>redisson-spring-boot-starter</artifactId>
	<version>3.13.1</version>
</dependency>

代码

	private final RedissonClient redissonClient;

    @GetMapping("/redisson/lock")
    public String redissonLock() {
        RLock lock = redissonClient.getLock("Csea");
        log.info("进入了方法");
        try {
            lock.lock(30, TimeUnit.SECONDS);
            log.info("拿到了锁");
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
        log.info("执行结束");
        return "执行结束";
    }

效果

启动8080,8081两个端口
Mysql、Redis、Redisson分布式锁
Mysql、Redis、Redisson分布式锁

JMeter压测

压测代码
@Service
@Slf4j
public class JmService {

    static Map<String, Integer> productMap = new HashMap<>(10);
    static Map<String, Integer> stockMap = new HashMap<>(10);
    static Map<String, String> orderMap = new HashMap<>();

    static {
        productMap.put("123456", 100000);
        stockMap.put("123456", 100000);
    }

    private String queryMap(String projectId) {
        return "商品限量:" + productMap.get(projectId) +
                "还剩下:" + stockMap.get(projectId) + " 份" +
                "下单用户数数量:" + orderMap.size();
    }

    public String querySecKillProductInfo(String projectId) {
        return this.queryMap(projectId);
    }


    @Autowired
    private RedissonClient redissonClient;

    public void secKillProject(String productId) {
        RLock lock = redissonClient.getLock("myLock");

        log.info("锁状态:{}", lock.isLocked());
        try {
            if (lock.isLocked()) {
                throw new RuntimeException("当前请求过多");
            }
            // 加锁
            lock.lock();

            // 如果库存为0 活动结束
            int stockNum = stockMap.get(productId);
            if (stockNum == 0) {
                throw new RuntimeException("商品已售空,活动结束");
            } else {
                // 模拟订单
                orderMap.put(String.valueOf(System.currentTimeMillis()), productId);
                // 扣库存
                stockNum -= 1;

                stockMap.put(productId, stockNum);
            }
        } catch (RuntimeException e) {
            e.printStackTrace();
        } finally {
            // 解锁
            lock.unlock();
        }
    }
}
	private final JmService jmService;

    /**
     * 利用Redisson进行秒杀压测 Demo
     *
     * @param productId
     * @return
     */
    @GetMapping("/redisson/query/{productId}")
    public String redissonQuery(@PathVariable String productId) {
        return jmService.querySecKillProductInfo(productId);
    }

    @GetMapping("/redisson/order/{productId}")
    public String redissonSkill(@PathVariable String productId) {
        jmService.secKillProject(productId);
        return jmService.querySecKillProductInfo(productId);
    }
配置

5000个线程同时访问
Mysql、Redis、Redisson分布式锁
Mysql、Redis、Redisson分布式锁

效果

Mysql、Redis、Redisson分布式锁
Mysql、Redis、Redisson分布式锁
Mysql、Redis、Redisson分布式锁

对比

方式 优点 缺点
数据库 实现简单,易理解 数据库压力大
Redis 易理解 不支持阻塞
Redisson 提供锁的方法,可以阻塞

本文地址:https://blog.csdn.net/qq_38494341/article/details/107359544