目前几乎很多大型网站及应用都是分布式部署的,分布式场景中的数据一致性问题一直是一个比较重要的话题。分布式的CAP理论告诉我们“任何一个分布式系统都无法同时满足一致性(Consistency)、可用性(Availability)和分区容错性(Partition tolerance),最多只能同时满足两项。”所以,很多系统在设计之初就要对这三者做出取舍。在互联网领域的绝大多数的场景中,都需要牺牲强一致性来换取系统的高可用性,系统往往只需要保证“最终一致性”,只要这个最终时间是在用户可以接受的范围内即可。在很多场景中,我们为了保证数据的最终一致性,需要很多的技术方案来支持,比如分布式事务、分布式锁等。

分布式锁与实现(二)—基于 zookeeper 实现-图片1

什么是分布式锁
分布式锁一般用在分布式系统或者多个应用中,用来控制同一任务是否执行或者任务的执行顺序。在项目中,部署了多个tomcat应用,在执行定时任务时就会遇到同一任务可能执行多次的情况,我们可以借助分布式锁,保证在同一时间只有一个tomcat应用执行了定时任务。

分布式锁有那些实现方案

1.使用数据库实现分布式锁(不推荐使用)
缺点:效率低,性能差、线程出现异常时,容易出现死锁。
2.使用redis实现分布式锁
缺点:锁的失效时间难控制、容易产生死锁、非阻塞式、不可重入。
3.使用zookeeper实现分布式锁
实现相对简单、可靠性强、使用临时节点,失效时间容易控制。
4.springcloud内置实现全局锁
比较冷门,但是效率最高,实现简单,失效时间容易控制。

应用场景
分布式锁主要用于在分布式环境中保护跨进程、跨主机、跨网络的共享资源实现互斥访问,以达到保证数据的一致性。

实现原理
利用Zookeeper来实现分布式锁,主要基于其临时(或临时有序)节点和watch机制。

架构分析

分布式锁与实现(二)—基于 zookeeper 实现-图片2

实现思路和流程
1、在zookeeper指定节点(locker)下创建临时顺序节点node_n。
2、获取locker下所有子节点children。
3、对子节点按节点自增序号从小到大排序。
4、判断本节点是不是第一个子节点,若是,则获取锁;若不是,则等待。
5、使用zookeeper感知节点的功能,对本节点的上一个节点进行感知。
6、当上一个节点被删除了,zookeeper会通知该线程,该线程就结束等待,并获取锁。
7、释放锁,并删除该临时节点。

具体实现代码
下面就具体使用java和zookeeper实现分布式锁,操作zookeeper使用的是apache提供的zookeeper的包。

分布式锁类

  1. import org.apache.zookeeper.*;
  2. import org.apache.zookeeper.data.Stat;
  3.  
  4. import java.io.IOException;
  5. import java.util.ArrayList;
  6. import java.util.Collections;
  7. import java.util.List;
  8. import java.util.concurrent.CountDownLatch;
  9. import java.util.concurrent.TimeUnit;
  10. import java.util.concurrent.locks.Condition;
  11. import java.util.concurrent.locks.Lock;
  12.  
  13. /**
  14. * Lock:Java的锁接口,并实现该接口的抽象方法
  15. * Watcher:zookeeper的节点感知接口,并实现process方法,当节点有改变时,会调用该方法
  16. */
  17. public class DistributedLock implements Lock, Watcher {
  18.  
  19.  
  20. private ZooKeeper zk = null;
  21. // 根节点
  22. private String ROOT_LOCK = "/locker";
  23. // 竞争的资源
  24. private String lockName;
  25. // 等待的前一个锁
  26. private String WAIT_LOCK;
  27. // 当前锁
  28. private String CURRENT_LOCK;
  29. // 同步计数器
  30. private CountDownLatch countDownLatch;
  31. private int sessionTimeout = 30000;
  32.  
  33. /**
  34. * 配置分布式锁
  35. * @param config 连接的url
  36. * @param lockName 竞争资源
  37. */
  38. public DistributedLock(String config, String lockName) {
  39. this.lockName = lockName;
  40. try {
  41. // 连接zookeeper
  42. zk = new ZooKeeper(config, sessionTimeout, this);
  43. Stat stat = zk.exists(ROOT_LOCK, false);
  44. if (stat == null) {
  45. // 如果根节点不存在,则创建根节点
  46. zk.create(ROOT_LOCK, new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
  47. }
  48. } catch (IOException e) {
  49. e.printStackTrace();
  50. } catch (InterruptedException e) {
  51. e.printStackTrace();
  52. } catch (KeeperException e) {
  53. e.printStackTrace();
  54. }
  55. }
  56.  
  57. // 节点感知器,感知到节点变化会调用该方法
  58. public void process(WatchedEvent event) {
  59. System.out.println("感知节点变化类型:"+event.getType().name());
  60. if (this.countDownLatch != null) {
  61. //如果同步计数器不为null,则减一
  62. this.countDownLatch.countDown();
  63. }
  64. }
  65.  
  66. public void lock() {
  67.  
  68. try {
  69. if (this.tryLock()) {
  70. System.out.println(Thread.currentThread().getName() + " " + lockName + "获得了锁");
  71. return;
  72. } else {
  73. // 等待锁
  74. waitForLock(WAIT_LOCK, sessionTimeout);
  75. }
  76. } catch (InterruptedException e) {
  77. e.printStackTrace();
  78. } catch (KeeperException e) {
  79. e.printStackTrace();
  80. }
  81. }
  82.  
  83. public boolean tryLock() {
  84. try {
  85. String splitStr = "_lock_";
  86. if (lockName.contains(splitStr)) {
  87. throw new RuntimeException("锁名有误");
  88. }
  89. // 创建临时有序节点
  90. CURRENT_LOCK = zk.create(ROOT_LOCK + "/" + lockName + splitStr, new byte[0],
  91. ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL_SEQUENTIAL);
  92. System.out.println(CURRENT_LOCK + " 已经创建");
  93. // 取所有子节点
  94. List subNodes = zk.getChildren(ROOT_LOCK, false);
  95. // 取出所有lockName的锁
  96. List lockObjects = new ArrayList();
  97. for (String node : subNodes) {
  98. String _node = node.split(splitStr)[0];
  99. if (_node.equals(lockName)) {
  100. lockObjects.add(node);
  101. }
  102. }
  103. Collections.sort(lockObjects);
  104. System.out.println(Thread.currentThread().getName() + " 的锁是 " + CURRENT_LOCK);
  105. // 若当前节点为最小节点,则获取锁成功
  106. if (CURRENT_LOCK.equals(ROOT_LOCK + "/" + lockObjects.get(0))) {
  107. return true;
  108. }
  109.  
  110. // 若不是最小节点,则找到自己的前一个节点
  111. String prevNode = CURRENT_LOCK.substring(CURRENT_LOCK.lastIndexOf("/") + 1);
  112. WAIT_LOCK = lockObjects.get(Collections.binarySearch(lockObjects, prevNode) - 1);
  113. } catch (InterruptedException e) {
  114. e.printStackTrace();
  115. } catch (KeeperException e) {
  116. e.printStackTrace();
  117. }
  118. return false;
  119. }
  120.  
  121. public boolean tryLock(long timeout, TimeUnit unit) {
  122. try {
  123. if (this.tryLock()) {
  124. return true;
  125. }
  126. return waitForLock(WAIT_LOCK, timeout);
  127. } catch (Exception e) {
  128. e.printStackTrace();
  129. }
  130. return false;
  131. }
  132.  
  133. // 等待锁
  134. private boolean waitForLock(String prev, long waitTime) throws KeeperException, InterruptedException {
  135.  
  136. //获取并感知上一个节点
  137. Stat stat = zk.exists(ROOT_LOCK + "/" + prev, true);
  138.  
  139. if (stat != null) {
  140. System.out.println(Thread.currentThread().getName() + "等待锁 " + ROOT_LOCK + "/" + prev);
  141. //初始化同步计数器,计数为1,当同步计数器为0,主线程才会向下执行
  142. this.countDownLatch = new CountDownLatch(1);
  143. // 计数等待,若等到前一个节点消失,则precess中进行countDown,停止等待,获取锁
  144. this.countDownLatch.await(waitTime, TimeUnit.MILLISECONDS);
  145. this.countDownLatch = null;
  146. System.out.println(Thread.currentThread().getName() + " 等到了锁");
  147. }
  148. return true;
  149. }
  150.  
  151. public void unlock() {
  152. try {
  153. System.out.println("释放锁 " + CURRENT_LOCK);
  154. zk.delete(CURRENT_LOCK, -1);
  155. CURRENT_LOCK = null;
  156. zk.close();
  157. } catch (InterruptedException e) {
  158. e.printStackTrace();
  159. } catch (KeeperException e) {
  160. e.printStackTrace();
  161. }
  162. }
  163.  
  164. public Condition newCondition() {
  165. return null;
  166. }
  167.  
  168.  
  169. public void lockInterruptibly() throws InterruptedException {
  170. this.lock();
  171. }
  172. }

测试类

  1. public class Test {
  2.  
  3. static int n = 500;
  4.  
  5. public static void secskill() {
  6. System.out.println(--n);
  7. }
  8.  
  9. public static void main(String[] args) {
  10.  
  11. Runnable runnable = new Runnable() {
  12. public void run() {
  13. DistributedLock lock = null;
  14. try {
  15. lock = new DistributedLock("127.0.0.1:2181", "node");
  16. lock.lock();
  17. secskill();
  18. System.out.println(Thread.currentThread().getName() + "正在运行");
  19. } finally {
  20. if (lock != null) {
  21. lock.unlock();
  22. }
  23. }
  24. }
  25. };
  26.  
  27. for (int i = 0; i < 10; i++) {
  28. Thread t = new Thread(runnable);
  29. t.start();
  30. }
  31. }
  32. }

本文已通过「原本」原创作品认证,转载请注明文章出处及链接。

Java最后更新:2022-11-6
夏日阳光
  • 本文由 夏日阳光 发表于 2019年10月3日
  • 本文为夏日阳光原创文章,转载请务必保留本文链接:https://www.pieruo.com/104.html
匿名

发表评论

匿名网友
:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:
确定

拖动滑块以完成验证
加载中...