锁是开发过程中十分常见的工具,在处理高并发请求的时候和订单数据的时候往往需要锁来帮助我们保证数据的安全。

一、分布式锁的场景

  • 场景1.前端点击太快,导致后端重复调用接口。两次调用一个接口,这样就会产生同一个请求执行了两次,而从用户的角度出发,他是因为太卡而点了两次,他的目标是执行一次请求。

  • 场景2.对于高并发场景,我们往往需要引入分布式缓存,来加快整个系统的响应速度。但是缓存是有失效机制的,如果某一时刻缓存失效,而此时有大量的请求过来,那么所有的请求会瞬间直接打到DB上,那么这么大的并发量,DB可能是扛不住的。那么这里需要引入一个保护机制。当发生“缓存击穿”的时候加锁,从而保护DB不被拖垮。

看完了上面的场景,其实分布式锁的场景一直在我们身边。说分布式锁之前,应该先说一下java提供的锁,比较能单机解决的并发问题,没必要引入分布式的解决方案。java提供了两种内置的锁的实现,一种是由JVM实现的synchronized和JDK提供的Lock,当你的应用是单机或者说单进程应用时,可以使用synchronized或Lock来实现锁。但是,当你的应用涉及到多机、多进程共同完成时,例如现在的互联网架构,一般都是分布式的RPC框架来支撑,那么这样你的Server有多个,由于负载均衡的路由规则随机,相同的请求可能会打到不同的Server上进行处理,那么这时候就需要一个全局锁来实现多个线程(不同的进程)之间的同步。实现全局的锁需要依赖一个第三方系统,此系统需要满足高可用、一致性比较强同时能应付高并发的请求。常见的处理办法有三种:数据库、缓存、分布式协调系统。数据库和缓存是比较常用的,但是分布式协调系统是不常用的。

二、数据库实现分布式锁

利用DB来实现分布式锁,有两种方案。两种方案各有好坏,但是总体效果都不是很好。但是实现还是比较简单的。

  1. 利用主键唯一规则:
    我们知道数据库是有唯一主键规则的,主键不能重复,对于重复的主键会抛出主键冲突异常。
    了解JDK reentrantlock的人都知道,reentrantlock是利用了OS的CAS特性实现的锁。主要是维护一个全局的状态,每次竞争锁都会CAS修改锁的状态,修改成功之后就占用了锁,失败的加入到同步队列中,等待唤醒。其实这和分布式锁实现方案基本是一致的,首先我们利用主键唯一规则,在争抢锁的时候向DB中写一条记录,这条记录主要包含锁的id、当前占用锁的线程名、重入的次数和创建时间等,如果插入成功表示当前线程获取到了锁,如果插入失败那么证明锁被其他人占用,等待一会儿继续争抢,直到争抢到或者超时为止。
    这里我主要写了一个简单的实现:
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;/*** 利用mysql实现可重入分布式锁*/
public class MysqlprimaryLock {private static Connection connection;static {try {Class.forName("com.mysql.jdbc.Driver");} catch (ClassNotFoundException e) {e.printStackTrace();}String url = "jdbc:mysql://10.0.0.212:3308/dbwww_lock?user=lock_admin&password=lock123";try {connection = DriverManager.getConnection(url);} catch (SQLException e) {e.printStackTrace();}}/*** 加锁* @param lockID*/public void lock(String lockID) {acquire(lockID);}/*** 获取锁* @param lockID* @return*/public boolean acquire(String lockID) {String sql = "insert into test_lock('id','count','thName','addtime') VALUES (?,?,?,?)";while (true) {try {PreparedStatement statement = connection.prepareStatement(sql);statement.setString(1, lockID);statement.setInt(2, 1);statement.setLong(1, System.currentTimeMillis());boolean ifsucess = statement.execute();//如果成功,那么就是获取到了锁if (ifsucess)return true;} catch (SQLException e) {e.printStackTrace();}try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}continue;}}/*** 超时获取锁* @param lockID* @param timeOuts* @return* @throws InterruptedException*/public boolean acquire(String lockID, long timeOuts) throws InterruptedException {String sql = "insert into test_lock('id','count','thName','addtime') VALUES (?,?,?,?)";long futureTime = System.currentTimeMillis() + timeOuts;long ranmain = timeOuts;long timerange = 500;while (true) {CountDownLatch latch = new CountDownLatch(1);try {PreparedStatement statement = connection.prepareStatement(sql);statement.setString(1, lockID);statement.setInt(2, 1);statement.setLong(1, System.currentTimeMillis());boolean ifsucess = statement.execute();//如果成功,那么就是获取到了锁if (ifsucess)return true;} catch (SQLException e) {e.printStackTrace();}latch.await(timerange, TimeUnit.MILLISECONDS);ranmain = futureTime - System.currentTimeMillis();if (ranmain <= 0)break;if (ranmain < timerange) {timerange = ranmain;}continue;}return false;}/*** 释放锁* @param lockID* @return* @throws SQLException*/public boolean unlock(String lockID) throws SQLException {String sql = "DELETE  from test_lock where id = ?";PreparedStatement statement = connection.prepareStatement(sql);statement.setString(1, lockID);boolean ifsucess = statement.execute();if (ifsucess)return true;return false;}
}

这里是利用主键冲突规则,加入了id’,‘count’,‘thName’,‘addtime’,count主要是为了重入计数,thName为了判断占用锁的线程,addtime是记录占用时间。上面代码没有实现重入的逻辑。重入主要实现思路是,在每次获取锁之前去取当前锁的信息,如果锁的线程是当前线程,那么更新锁的count+1,并且执行锁之后的逻辑。如果不是当前锁,那么进行重试。释放的时候也要进行count-1,最后减到0时,删除锁标识释放锁。

  • 优点:实现简单

  • 缺点:没有超时保护机制,mysql存在单点,并发量大的时候请求量太大、没有线程唤醒机制,用异常去控制逻辑多少优点恶心。

  • 对于超时保护:如果可能,可以采用定时任务去扫描超过一定阈值的锁,并删除。但是也会存在,锁住的任务执行时间很长,删除锁会导致并发问题。所以需要对超时时间有一个很好的预估。

  • 对于单点问题:有条件可以搞一个主从,但是为了一个锁来搞一个主从是不是优点浪费?同时主从切换的时候系统不可用,这也是一个问题。

  • 并发量大的时候请求量太大:因为这种实现方式是没有锁的唤醒机制的,不像reentrantlock在同步队列中的节点,可以通过唤醒来避免多次的循环请求。但是分布式环境数据库这种锁的实现是不能做到唤醒的。所以只能将获取锁的时间间隔调高,避免死循环给系统和DB带来的巨大压力。这样也牺牲了系统的吞吐量,因为总会有一定的间隔锁是空闲的。

用异常去控制逻辑多少有点恶心:就不说了,每次失败都抛异常…

  1. 利用Mysql行锁的特性:

Mysql是有表锁、页锁和行锁的机制的,可以利用这个机制来实现锁。这里尽量使用行锁,它的吞吐量是最高的。

/*** 超时获取锁* @param lockID* @param timeOuts* @return* @throws InterruptedException*/
public boolean acquireByUpdate(String lockID, long timeOuts) throws InterruptedException, SQLException {String sql = "SELECT id from test_lock where id = ? for UPDATE ";long futureTime = System.currentTimeMillis() + timeOuts;long ranmain = timeOuts;long timerange = 500;connection.setAutoCommit(false);while (true) {CountDownLatch latch = new CountDownLatch(1);try {PreparedStatement statement = connection.prepareStatement(sql);statement.setString(1, lockID);statement.setInt(2, 1);statement.setLong(1, System.currentTimeMillis());boolean ifsucess = statement.execute();//如果成功,那么就是获取到了锁if (ifsucess)return true;} catch (SQLException e) {e.printStackTrace();}latch.await(timerange, TimeUnit.MILLISECONDS);ranmain = futureTime - System.currentTimeMillis();if (ranmain <= 0)break;if (ranmain < timerange) {timerange = ranmain;}continue;}return false;
}
/*** 释放锁* @param lockID* @return* @throws SQLException*/
public void unlockforUpdtate(String lockID) throws SQLException {connection.commit();
}

利用for update加显式的行锁,这样就能利用这个行级的排他锁来实现分布式锁了,同时unlock的时候只要释放commit这个事务,就能达到释放锁的目的。

  • 优点:实现简单

  • 缺点:连接池爆满和事务超时的问题单点的问题,单点问题,行锁升级为表锁的问题,并发量大的时候请求量太大、没有线程唤醒机制。

  • 连接池爆满和事务超时的问题单点的问题:利用事务进行加锁的时候,query需要占用数据库连接,在行锁的时候连接不释放,这就会导致连接池爆满。同时由于事务是有超时时间的,过了超时时间自动回滚,会导致锁的释放,这个超时时间要把控好。

  • 对于单点问题:同上。

  • 并发量大的时候请求量太大:同上。

  • 行锁升级为表锁的问题:Mysql行锁默认需要走索引,如果不走索引会导致锁表,如果可以,在sql中可以强制指定索引。

三、缓存分布式锁(redis、memcached)

缓存实现分布式锁还是比较常见的,因为缓存比较轻量,并且缓存的响应快、吞吐高。最重要的是还有自动失效的机制来保证锁一定能释放。`缓存的分布式锁主要通过Redis实现,当然其他的缓存也是可以的。

  1. 基于SetNX实现:

setNX是Redis提供的一个原子操作,如果指定key存在,那么setNX失败,如果不存在会进行Set操作并返回成功。我们可以利用这个来实现一个分布式的锁,主要思路就是,set成功表示获取锁,set失败表示获取失败,失败后需要重试。
具体看下伪代码:

这种方式也是有优点和缺点:

  • 优点:实现简单,吞吐量十分客观,对于高并发情况应付自如,自带超时保护,对于网络抖动的情况也可以利用超时删除策略保证不会阻塞所有流程。

  • 缺点:单点问题、没有线程唤醒机制、网络抖动可能会引起锁删除失败。

  • 对单点问题:因为redis一般都是单实例使用,那么对于单点问题,可以做一个主从。当然主从切换的时候也是不可用的,因为主从同步是异步的,可能会并发问题。如果对于主从还是不能保证可靠性的话,可以上Redis集群,对于Redis集群,因为使用了类一致性Hash算法,虽然不能避免节点下线的并发问题(当前的任务没有执行完,其他任务就开始执行),但是能保证Redis是可用的。可用性的问题是出了问题之后的备选方案,如果我们系统天天都出问题还玩毛啊,对于突发情况牺牲一两个请求还是没问题的。

  • 对于线程唤醒机制:分布式锁大多都是这样轮训获取锁的,所以控制住你的重试频率,也不会导致负载特别高的。可能就是吞吐量低点而已。

  • 对于锁删除失败:分布式锁基本都有这个问题,可以对key设置失效时间。这个超时时间需要把控好,过大那么系统吞吐量低,很容易导致超时。如果过小那么会有并发问题,部分耗时时间比较长的任务就要遭殃了。

关于redis作为分布式锁是否安全的一篇好文章

public Map<String, List<Catelog2Vo>> getCatalogJsonFromDbWithRedisLock() {//1、占分布式锁。去redis占坑String uuid = UUID.randomUUID().toString();Boolean lock=redisTemplate.opsForValue().setIfAbsent("lock",uuid,300,TimeUnit.SECONDS);if(lock){System.out.println("获取分布式锁成功...");//加锁成执行业务//2、设置过期时间,必须和加锁是同步的,原子的//redisTemplate.expire("lock",30,TimeUnit.SECONDS);Map<String, List<Catelog2Vo>> dataFromDb;try{dataFromDb = getDataFromDb();}finally {//lua脚本String script = "if redis.call('get', KEYS[1]) == ARGV[1] thenreturn redis.call('del', KEYS[1]) else return 0 end";//删除锁Long lock1 = redisTemplate.execute(new DefaultRedisScript<Long>(script, Long.class)
, Arrays.asList("lock"), uuid);}   return dataFromDb;}else{//加锁失败..synchronized ()//休眠100msSystem.out.println("获取锁失败..重试");try{Thread.sleep(200);}catch(Exception e){}
return getCatalogJsonFromDbWithRedisLock();//自旋的方式
}
}
  1. 使用Redisson完成分布式锁
    Redisson 是架设在Redis 基础上的一个 Java 驻内存数据网格(In-Memory Data Grid)。充分的利用了 Redis 键值数据库提供的一系列优势,基于Java 实用工具包中常用接口,为使用者提供了一系列具有分布式特性的常用工具类。使得原本作为协调单机多线程并发程序的工具包获得了协调分布式多机多线程并发系统的能力,大大降低了设计和研发大规模分布式系统的难度。同时结合各富特色的分布式服务,更进一步简化了分布式环境中程序相互之间的协作。
    官方文档

更详细的关于redis实现分布式锁的文章

四、基于Zookeeper的分布式锁

Zookeeper是一个分布式一致性协调框架,主要可以实现选主、配置管理和分布式锁等常用功能,因为Zookeeper的写入都是顺序的,在一个节点创建之后,其他请求再次创建便会失败,同时可以对这个节点进行Watch,如果节点删除会通知其他节点抢占锁。Zookeeper实现分布式锁虽然是比较重量级的,但实现的锁功能十分健全,由于Zookeeper本身需要维护自己的一致性,所以性能上较Redis还是有一定差距的。

五、对比

  • Mysql实现比较简单,不需要引入第三个应用,但实现多少有些重,性能不是很好。
  • Redis的话实现比较简单,同时性能很好,引入集群可以提高可用性。同时定期失效的机制可以解决因网络抖动锁删除失败的问题,所以我比较倾向Redis实现。
  • Zookeeper实现是有些重的,同时我们还需要维护Zookeeper集群,实现起来还是比较复杂的,实现不好的话还会引起“羊群效应”。如果不是原有系统就依赖Zookeeper,同时压力不大的情况下。一般不使用Zookeeper实现分布式锁。

参考文章

分布式锁的3种实现(数据库、缓存[redis]、Zookeeper)相关推荐

  1. (转)分布式锁的几种使用方式(redis、zookeeper、数据库)

    https://blog.csdn.net/u010963948/article/details/79006572 转载于:https://www.cnblogs.com/Andrew520/p/10 ...

  2. memcached 分布式锁 java_分布式锁的三种实现方式

    分布式锁的三种实现方式 一.zookeeper 1.实现原理: 基于zookeeper瞬时有序节点实现的分布式锁,其主要逻辑如下(该图来自于IBM网站).大致思想即为:每个客户端对某个功能加锁时,在z ...

  3. 分布式锁的实现方式——ACID数据库、缓存或者是zk

    针对分布式锁的实现,目前比较常用的有以下几种方案: 基于数据库实现分布式锁 基于缓存(redis,memcached,tair)实现分布式锁 基于Zookeeper实现分布式锁 在分析这几种实现方案之 ...

  4. 分布式锁的几种实现方式

    目前几乎很多大型网站及应用都是分布式部署的,分布式场景中的数据一致性问题一直是一个比较重要的话题.分布式的CAP理论告诉我们"任何一个分布式系统都无法同时满足一致性(Consistency) ...

  5. 搞懂分布式技术16:浅谈分布式锁的几种方案

    搞懂分布式技术16:浅谈分布式锁的几种方案 前言 随着互联网技术的不断发展,数据量的不断增加,业务逻辑日趋复杂,在这种背景下,传统的集中式系统已经无法满足我们的业务需求,分布式系统被应用在更多的场景, ...

  6. 分布式锁的三种实现方式_分布式锁的几种实现方式~

    目前几乎很多大型网站及应用都是分布式部署的,分布式场景中的数据一致性问题一直是一个比较重要的话题.分布式的CAP理论告诉我们"任何一个分布式系统都无法同时满足一致性(Consistency) ...

  7. 分布式锁的三种实现方式_分布式锁的多种实现方式

    目前几乎很多大型网站及应用都是分布式部署的,分布式场景中的数据一致性问题一直是一个比较重要的话题.分布式的CAP理论告诉我们"任何一个分布式系统都无法同时满足一致性(Consistency) ...

  8. 分布式锁的几种实现方式~

    目前几乎很多大型网站及应用都是分布式部署的,分布式场景中的数据一致性问题一直是一个比较重要的话题.分布式的CAP理论告诉我们"任何一个分布式系统都无法同时满足一致性(Consistency) ...

  9. 分布式锁的几种实现方式(转)

    出处:https://www.cnblogs.com/austinspark-jessylu/p/8043726.html 目前几乎很多大型网站及应用都是分布式部署的,分布式场景中的数据一致性问题一直 ...

  10. 【2020尚硅谷Java大厂面试题第三季 04】Redis 9种数据类型使用场景,分布式锁演变步骤,lua脚本,redis事务,Redisson,Redis内存占用,删除策略,内存淘汰策略,手写LRU

    1.安装redis6.0.8 2023 02 02 为:redis-7.0.8.tar.gz 2.redis传统五大数据类型的落地应用 3.知道分布式锁吗?有哪些实现方案?你谈谈对redis分布式锁的 ...

最新文章

  1. 新一届最强预训练模型上榜,出于BERT而胜于BERT
  2. wireshark从入门到精通(协议排错安全篇)4
  3. python如何读取一个文件夹下的多个文件(夹)?
  4. #再一次用construct2做游戏
  5. 寒冷的高纬度——我的梦开始的地方
  6. 上采样,下采样,过采样,欠采样的区别
  7. cocos2dx游戏开发——微信打飞机学习笔记(五)——BackgroundLayer的搭建
  8. EMLOG SSL插件 一键开启/关闭ssl无需操作数据库
  9. java继承时父类常量覆盖吗_Java父类继承中的static和final用法
  10. 一套组合拳,打造一款 IDEA 护眼方案
  11. PostgreSQL最终获得存储过程
  12. 国外大神一张图学会python-学习Python不得不关注和学习的国外大神博客
  13. 人脸检测(十二)--DDFD算法
  14. Go语言学习路线图 初阶+中阶+高阶
  15. pymysql executemany()函数
  16. uniapp动态修改元素节点样式
  17. 移动应用发展现状_移动应用开发平台:现状和趋势
  18. 零起点学习Linux系列培训视频 [共61课] (万分感谢寒冰老师分享!)
  19. cad工具箱详细讲解_AutoCAD学院派工具箱(XCAD)使用教程(图文解说)
  20. ftp服务器vsftpd配置文件,vsftpd配置文件详解及ftp服务器搭建

热门文章

  1. windows 7 memcached报failed to install service or service already installed的解决方案
  2. Linux - 磁盘操作
  3. React-Native 之 项目实战(五)
  4. linux下echo命令详解(转)
  5. js中直接对字符串转义-用于solr ulr 关键词转义
  6. DDoS攻防战(三):ip黑白名单防火墙frdev的原理与实现
  7. 2.1 Java程序的构成
  8. Android文件系统的结构
  9. android TextView 的垂直滚动
  10. 编程方法学4:计算机科学发展简史