redis分布式锁

概述

在需求开发中,经常会有类似需求:多个数据库操作之间要保证原子性。

如果是单机情况下,可以通过加锁实现操作的原子性,例如使用synchronized关键字,但是如果是分布式环境下,即使加了synchronized关键字,由于存在多个实例,每个实例间的变量并不共享,因此无法保证操作的原子性。

这种情况下,就需要通过分布式锁进行并发控制。

分布式锁需要满足以下条件:

1、对所有实例可见

2、具备超时自动释放的能力,避免由于获取锁的实例宕机导致产生死锁问题

实现

分布式锁有几种实现方式:redis、数据库、zookeeper

如今最流行的方式是通过redis实现分布式锁。

示例:假设数据库存储班级信息,每个学生对应一条记录

|----|----|------|---------|
| id | no | name | monitor |
| 主键 | 学号 | 姓名 | 是否为班长 |

现有交易 competeForMonitor(),需要实现效果为:首个调用交易的学生即为班长

代码实现如下:

java 复制代码
competeForMonitor(int id) {
    // 1、查询班长人数
    int num = classroom.getMonitorMum();    

    // 2、如果当前不存在,则更新为班长
    if (num == 0) {
        classroom.updateById(id);
    }
}

上述代码在没有并发的情况下能够正常执行,但是若存在并发场景,则会产生以下问题:

如果多个线程同时执行了第一步,获取到当前班长数量为0,则会同时执行第二步,将自己设置为班长,与原需求不符。

通过redis的setnx(set not exists如果不存在则设置key)命令可以实现操作的原子性。

由于redis本身为单线程操作,因此当有多个请求并发执行时,redis也能按顺序逐条执行,不会产生并发问题。

java 复制代码
competeForMonitor(int id) {
    // 如果获取锁失败,则自旋尝试获取锁
    while(!stringRedisTemplate.opsForValue().setIfAbsent("lock", "value")) {
    }
    
    // 1、查询班长人数
    int num = classroom.getMonitorMum();    

    // 2、如果当前不存在,则更新为班长
    if (num == 0) {
        classroom.updateById(id);
    }
    // 释放锁
    stringRedisTemplate.delete("lock");
}

问题

上述代码可以实现最基本的分布式锁功能,即加锁和解锁操作。

然而,实际情况可能会有突发问题导致代码运行出现错误。比如,如果A实例获取到锁后挂了,锁未被释放,导致其他实例无法获取锁,影响正常业务,应该如何处理。

给锁加上过期时间?如果中间查询和更新操作实际耗时不确定,过期时间怎么确定?如果实际操作时间大于过期时间,锁被释放了,无法满足原子性怎么办?

相关推荐
倔强的石头_16 小时前
kingbase备份与恢复实战(二)—— sys_dump库级逻辑备份与恢复(Windows详细步骤)
数据库
jiayou642 天前
KingbaseES 实战:深度解析数据库对象访问权限管理
数据库
李广坤3 天前
MySQL 大表字段变更实践(改名 + 改类型 + 改长度)
数据库
初次攀爬者4 天前
ZooKeeper 实现分布式锁的两种方式
分布式·后端·zookeeper
爱可生开源社区4 天前
2026 年,优秀的 DBA 需要具备哪些素质?
数据库·人工智能·dba
随逸1774 天前
《从零搭建NestJS项目》
数据库·typescript
加号35 天前
windows系统下mysql多源数据库同步部署
数据库·windows·mysql
シ風箏5 天前
MySQL【部署 04】Docker部署 MySQL8.0.32 版本(网盘镜像及启动命令分享)
数据库·mysql·docker
李慕婉学姐5 天前
Springboot智慧社区系统设计与开发6n99s526(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
数据库·spring boot·后端
百锦再5 天前
Django实现接口token检测的实现方案
数据库·python·django·sqlite·flask·fastapi·pip