<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          ZooKeeper分布式鎖的實現(xiàn)

          共 17332字,需瀏覽 35分鐘

           ·

          2021-07-05 13:53

          點擊上方藍色字體,選擇“標星公眾號”

          優(yōu)質(zhì)文章,第一時間送達

            作者 |  IT王小二

          來源 |  urlify.cn/EvQ7Fv

          在分布式的情況下,sychornized 和 Lock 已經(jīng)不能滿足我們的要求了,那么就需要使用第三方的鎖了,這里我們就使用 ZooKeeper 來實現(xiàn)一個分布式鎖。

          一、分布式鎖方案比較

          方案實現(xiàn)思路優(yōu)點缺點
          利用 MySQL 的實現(xiàn)方案利用數(shù)據(jù)庫自身提供的鎖機制實現(xiàn),要求數(shù)據(jù)庫支持行級鎖實現(xiàn)簡單性能差,無法適應(yīng)高并發(fā)場景;容易出現(xiàn)死鎖的情況;無法優(yōu)雅的實現(xiàn)阻塞式鎖
          利用 Redis 的實現(xiàn)方案使用 Setnx 和 lua 腳本機制實現(xiàn),保證對緩存操作序列的原子性性能好實現(xiàn)相對復(fù)雜,有可能出現(xiàn)死鎖;無法優(yōu)雅的實現(xiàn)阻塞式鎖
          利用 ZooKeeper 的實現(xiàn)方案基于 ZooKeeper 節(jié)點特性及 watch 機制實現(xiàn)性能好,穩(wěn)定可靠性高,能較好地實現(xiàn)阻塞式鎖實現(xiàn)相對復(fù)雜

          二、ZooKeeper實現(xiàn)分布式鎖

          這里使用 ZooKeeper 來實現(xiàn)分布式鎖,以50個并發(fā)請求來獲取訂單編號為例,描述兩種方案,第一種為基礎(chǔ)實現(xiàn),第二種在第一種基礎(chǔ)上進行了優(yōu)化。

          1. 方案一

          流程描述:

          具體代碼:

          OrderNumGenerator:

          /**
           * @Author SunnyBear
           * @Description 生成隨機訂單號
           */
          public class OrderNumGenerator {

              private static long count = 0;

              /**
               * 使用日期加數(shù)值拼接成訂單號
               */
              public String getOrderNumber() throws Exception {
                  String date = DateTimeFormatter.ofPattern("yyyyMMddHHmmss").format(LocalDateTime.now());
                  String number = new DecimalFormat("000000").format(count++);
                  return date + number;
              }
          }

          Lock:

          /**
           * @Author SunnyBear
           * @Description 自定義鎖接口
           */
          public interface Lock {

              /**
               * 獲取鎖
               */
              public void getLock();

              /**
               * 釋放鎖
               */
              public void unLock();
          }

          AbstractLock:

          /**
           * @Author SunnyBear
           * @Description 定義一個模板,具體的方法由子類來實現(xiàn)
           */
          public abstract class AbstractLock implements Lock {

              /**
               * 獲取鎖
               */
              @Override
              public void getLock() {

                  if (tryLock()) {
                      System.out.println("--------獲取到了自定義Lock鎖的資源--------");
                  } else {
                      // 沒拿到鎖則阻塞,等待拿鎖
                      waitLock();
                      getLock();
                  }

              }

              /**
               * 嘗試獲取鎖,如果拿到了鎖返回true,沒有拿到則返回false
               */
              public abstract boolean tryLock();

              /**
               * 阻塞,等待獲取鎖
               */
              public abstract void waitLock();
          }

          ZooKeeperAbstractLock:

          /**
           * @Author SunnyBear
           * @Description 定義需要的服務(wù)連接
           */
          public abstract class ZooKeeperAbstractLock extends AbstractLock {

              private static final String SERVER_ADDR = "192.168.182.130:2181,192.168.182.131:2181,192.168.182.132:2181";

              protected ZkClient zkClient = new ZkClient(SERVER_ADDR);

              protected static final String PATH = "/lock";
          }

          ZooKeeperDistrbuteLock:

          /**
           * @Author SunnyBear
           * @Description 真正實現(xiàn)鎖的細節(jié)
           */
          public class ZooKeeperDistrbuteLock extends ZooKeeperAbstractLock {
              private CountDownLatch countDownLatch = null;

              /**
               * 嘗試拿鎖
               */
              @Override
              public boolean tryLock() {
                  try {
                      // 創(chuàng)建臨時節(jié)點
                      zkClient.createEphemeral(PATH);
                      return true;
                  } catch (Exception e) {
                      // 創(chuàng)建失敗報異常
                      return false;
                  }
              }

              /**
               * 阻塞,等待獲取鎖
               */
              @Override
              public void waitLock() {
                  // 創(chuàng)建監(jiān)聽
                  IZkDataListener iZkDataListener = new IZkDataListener() {
                      @Override
                      public void handleDataChange(String s, Object o) throws Exception {

                      }

                      @Override
                      public void handleDataDeleted(String s) throws Exception {
                          // 釋放鎖,刪除節(jié)點時喚醒等待的線程
                          if (countDownLatch != null) {
                              countDownLatch.countDown();
                          }
                      }
                  };

                  // 注冊監(jiān)聽
                  zkClient.subscribeDataChanges(PATH, iZkDataListener);

                  // 節(jié)點存在時,等待節(jié)點刪除喚醒
                  if (zkClient.exists(PATH)) {
                      countDownLatch = new CountDownLatch(1);
                      try {
                          countDownLatch.await();
                      } catch (InterruptedException e) {
                          e.printStackTrace();
                      }
                  }

                  // 刪除監(jiān)聽
                  zkClient.unsubscribeDataChanges(PATH, iZkDataListener);
              }

              /**
               * 釋放鎖
               */
              @Override
              public void unLock() {
                  if (zkClient != null) {
                      System.out.println("釋放鎖資源");
                      zkClient.delete(PATH);
                      zkClient.close();
                  }
              }
          }

          測試效果:使用50個線程來并發(fā)測試ZooKeeper實現(xiàn)的分布式鎖

          /**
           * @Author SunnyBear
           * @Description 使用50個線程來并發(fā)測試ZooKeeper實現(xiàn)的分布式鎖
           */
          public class OrderService {

              private static class OrderNumGeneratorService implements Runnable {

                  private OrderNumGenerator orderNumGenerator = new OrderNumGenerator();;
                  private Lock lock = new ZooKeeperDistrbuteLock();

                  @Override
                  public void run() {
                      lock.getLock();
                      try {
                          System.out.println(Thread.currentThread().getName() + ", 生成訂單編號:"  + orderNumGenerator.getOrderNumber());
                      } catch (Exception e) {
                          e.printStackTrace();
                      } finally {
                          lock.unLock();
                      }
                  }
              }

              public static void main(String[] args) {
                  System.out.println("----------生成唯一訂單號----------");
                  for (int i = 0; i < 50; i++) {
                      new Thread(new OrderNumGeneratorService()).start();
                  }
              }
          }

          2. 方案二

          方案二在方案一的基礎(chǔ)上進行優(yōu)化,避免產(chǎn)生“羊群效應(yīng)”,方案一一旦臨時節(jié)點刪除,釋放鎖,那么其他在監(jiān)聽這個節(jié)點變化的線程,就會去競爭鎖,同時訪問 ZooKeeper,那么怎么更好的避免各線程的競爭現(xiàn)象呢,就是使用臨時順序節(jié)點,臨時順序節(jié)點排序,每個臨時順序節(jié)點只監(jiān)聽它本身的前一個節(jié)點變化。

          流程描述:

          具體代碼

          具體只需要將方案一中的 ZooKeeperDistrbuteLock 改變,增加一個 ZooKeeperDistrbuteLock2,測試代碼中使用 ZooKeeperDistrbuteLock2 即可測試,其他代碼都不需要改變。

          /**
           * @Author SunnyBear
           * @Description 真正實現(xiàn)鎖的細節(jié)
           */
          public class ZooKeeperDistrbuteLock2 extends ZooKeeperAbstractLock {

              private CountDownLatch countDownLatch = null;
              /**
               * 當前請求節(jié)點的前一個節(jié)點
               */
              private String beforePath;
              /**
               * 當前請求的節(jié)點
               */
              private String currentPath;

              public ZooKeeperDistrbuteLock2() {
                  if (!zkClient.exists(PATH)) {
                      // 創(chuàng)建持久節(jié)點,保存臨時順序節(jié)點
                      zkClient.createPersistent(PATH);
                  }
              }

              @Override
              public boolean tryLock() {
                  // 如果currentPath為空則為第一次嘗試拿鎖,第一次拿鎖賦值currentPath
                  if (currentPath == null || currentPath.length() == 0) {
                      // 在指定的持久節(jié)點下創(chuàng)建臨時順序節(jié)點
                      currentPath = zkClient.createEphemeralSequential(PATH + "/""lock");
                  }
                  // 獲取所有臨時節(jié)點并排序,例如:000044
                  List<String> childrenList = zkClient.getChildren(PATH);
                  Collections.sort(childrenList);

                  if (currentPath.equals(PATH + "/" + childrenList.get(0))) {
                      // 如果當前節(jié)點在所有節(jié)點中排名第一則獲取鎖成功
                      return true;
                  } else {
                      int wz = Collections.binarySearch(childrenList, currentPath.substring(6));
                      beforePath = PATH + "/" + childrenList.get(wz - 1);
                  }
                  return false;
              }

              @Override
              public void waitLock() {
                  // 創(chuàng)建監(jiān)聽
                  IZkDataListener iZkDataListener = new IZkDataListener() {
                      @Override
                      public void handleDataChange(String s, Object o) throws Exception {

                      }

                      @Override
                      public void handleDataDeleted(String s) throws Exception {
                          // 釋放鎖,刪除節(jié)點時喚醒等待的線程
                          if (countDownLatch != null) {
                              countDownLatch.countDown();
                          }
                      }
                  };

                  // 注冊監(jiān)聽,這里是給排在當前節(jié)點前面的節(jié)點增加(刪除數(shù)據(jù)的)監(jiān)聽,本質(zhì)是啟動另外一個線程去監(jiān)聽前置節(jié)點
                  zkClient.subscribeDataChanges(beforePath, iZkDataListener);

                  // 前置節(jié)點存在時,等待前置節(jié)點刪除喚醒
                  if (zkClient.exists(beforePath)) {
                      countDownLatch = new CountDownLatch(1);
                      try {
                          countDownLatch.await();
                      } catch (InterruptedException e) {
                          e.printStackTrace();
                      }
                  }

                  // 刪除對前置節(jié)點的監(jiān)聽
                  zkClient.unsubscribeDataChanges(beforePath, iZkDataListener);
              }

              /**
               * 釋放鎖
               */
              @Override
              public void unLock() {
                  if (zkClient != null) {
                      System.out.println("釋放鎖資源");
                      zkClient.delete(currentPath);
                      zkClient.close();
                  }
              }
          }








          瀏覽 50
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  啪视频在线 | 污污免费网站 | 乱伦日韩欧美 | 欧美黄色三级网站 | 成人深爱激情婷婷 |