首頁 > 軟體

springboot+zookeeper實現分散式鎖的範例程式碼

2022-03-21 19:01:13

InterProcessMutex內部實現了zookeeper分散式鎖的機制,所以接下來我們嘗試使用這個工具來為我們的業務加上分散式鎖處理的功能

zookeeper分散式鎖的特點:1、分散式 2、公平鎖 3、可重入

依賴

<dependency>
   <groupId>org.apache.zookeeper</groupId>
   <artifactId>zookeeper</artifactId>
   <version>3.4.10</version>
</dependency>
<!-- zookeeper 使用者端 -->
<dependency>
   <groupId>org.apache.curator</groupId>
   <artifactId>curator-framework</artifactId>
   <version>2.12.0</version>
</dependency>
<dependency>
   <groupId>org.apache.curator</groupId>
   <artifactId>curator-recipes</artifactId>
   <version>2.12.0</version>
</dependency>
<!-- lombok -->
<dependency>
   <groupId>org.projectlombok</groupId>
   <artifactId>lombok</artifactId>
   <version>1.18.16</version>
   <scope>provided</scope>
</dependency>

本地封裝

這個工具類主要封裝CuratorFramework這個client(連線Zookeeper)

@Slf4j
public class CuratorClientUtil {
    private String zookeeperServer;

    @Getter
    private CuratorFramework client;

    public CuratorClientUtil(String zookeeperServer) {
        this.zookeeperServer = zookeeperServer;
    }

  	// 建立CuratorFrameworkFactory並且啟動
    public void init() {
       // 重試策略,等待1s,最大重試3次
        RetryPolicy retryPolicy = new ExponentialBackoffRetry(1000,3);
        this.client = CuratorFrameworkFactory.builder()
                .connectString(zookeeperServer)
                .sessionTimeoutMs(5000)
                .connectionTimeoutMs(5000)
                .retryPolicy(retryPolicy)
                .build();
        this.client.start();
    }

   // 容器關閉,CuratorFrameworkFactory關閉
    public void destroy() {
        try {
            if (Objects.nonNull(getClient())) {
                getClient().close();
            }
        } catch (Exception e) {
            log.info("CuratorFramework close error=>{}", e.getMessage());
        }
    }
}

設定

@Configuration
public class CuratorConfigration {
    @Value("${zookeeper.server}")
    private String zookeeperServer;
    // 注入時,指定initMethod和destroyMethod
    @Bean(initMethod = "init", destroyMethod = "destroy")
    public CuratorClientUtil curatorClientUtil() {
        CuratorClientUtil clientUtil = new CuratorClientUtil(zookeeperServer);
        return clientUtil;
    }
}

測試程式碼

模擬不同使用者端的請求

@Slf4j
@RestController
@RequestMapping("/test")
public class TestController {
    // 注入client工具類
    @Autowired
    private CuratorClientUtil curatorClientUtil;
    // 在zookeeper的/rootLock節點下建立鎖對應的臨時有序節點
    private String rootLock = "/rootLock";

    @GetMapping("/testLock")
    public Object testLock() throws Exception {
        // 獲取當前執行緒的名字,方便觀察那些執行緒在獲取鎖
        String threadName = Thread.currentThread().getName();
        InterProcessMutex mutex = new InterProcessMutex(curatorClientUtil.getClient(), rootLock);
        try {
            log.info("{}---獲取鎖start", threadName);
            // 嘗試獲取鎖,最長等待3s,超時放棄獲取
            boolean lockFlag = mutex.acquire(3000, TimeUnit.SECONDS);
            // 獲取鎖成功,進行業務處理
            if (lockFlag) {
                log.info("{}---獲取鎖success", threadName);
                // 模擬業務處理,時間為3s
                Thread.sleep(3000);
            } else {
                log.info("{}---獲取鎖fail", threadName);
            }
        } catch (Exception e) {
            log.info("{}---獲取鎖異常", threadName);
        } finally {
            // 業務處理完成,釋放鎖,喚醒比當前執行緒建立的節點序號大(最靠近)的執行緒獲取鎖
            mutex.release();
            log.info("{}---鎖release", threadName);
        }
        return "執行緒:" + threadName + "執行完成";
    }
}

JMeter測試

我們使用JMeter模擬100個使用者端同時並行的存取 localhost:8081/test/testLock,相當於100個使用者端爭搶分散式鎖,結果如圖右上角所示,100個請求花了5分6s,每個執行緒獲取到鎖後業務處理3s,100個執行緒理想時間為300s(Thread.sleep(3000)),所以執行時間符合。

zookeeper每個執行緒在/rooLock節點下建立的臨時有序節點如下圖,由於是臨時的,所以執行緒釋放鎖後這些節點也會刪除

100個執行緒程式紀錄檔列印

關於InterProcessMutex內部如何實現zookeeper分散式鎖,請看我寫的這篇文章:在這裡

到此這篇關於springboot+zookeeper實現分散式鎖的範例程式碼的文章就介紹到這了,更多相關springboot zookeeper分散式鎖內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com