基于Zookeeper實(shí)現(xiàn)服務(wù)注冊(cè)和服務(wù)發(fā)現(xiàn)功能
前言
無論是采用SOA還是微服務(wù)架構(gòu),都需要使用服務(wù)注冊(cè)和服務(wù)發(fā)現(xiàn)組件。我剛開始接觸 Dubbo 時(shí)一直對(duì)服務(wù)注冊(cè)/發(fā)現(xiàn)以及 Zookeeper 的作用感到困惑,現(xiàn)在看來是因?yàn)閷?duì)分布式系統(tǒng)的理解不夠深入,對(duì) Dubbo 和 Zookeeper 的工作原理不夠清楚。
本文將基于 Zookeeper 實(shí)現(xiàn)服務(wù)注冊(cè)和服務(wù)發(fā)現(xiàn)功能,如果跟我一樣有同樣的困惑,希望可以通過本文了解其他組件如何使用 Zookeeper 作為注冊(cè)中心的工作原理。
聲明
文章中所提供的代碼僅供參考,旨在幫助缺乏基礎(chǔ)知識(shí)的開發(fā)人員更好地理解服務(wù)注冊(cè)和服務(wù)發(fā)現(xiàn)的概念。請(qǐng)注意,這些代碼并不適用于實(shí)際應(yīng)用中。
前置知識(shí)
服務(wù)注冊(cè)和發(fā)現(xiàn)
在SOA或微服務(wù)架構(gòu)中,由于存在大量的服務(wù)以及可能的相互調(diào)用,為了更有效地管理這些服務(wù),我們通常需要引入一個(gè)統(tǒng)一的地方,即注冊(cè)中心,來集中管理它們,而注冊(cè)中心最基本的功能就是服務(wù)注冊(cè)/發(fā)現(xiàn)。
- 服務(wù)注冊(cè):將該服務(wù)實(shí)例的元數(shù)據(jù)(如IP地址、端口號(hào)、健康狀態(tài)等)注冊(cè)到注冊(cè)中心,這樣其他服務(wù)或客戶端可以發(fā)現(xiàn)和使用該服務(wù)。
- 服務(wù)發(fā)現(xiàn):當(dāng)一個(gè)服務(wù)需要調(diào)用別的服務(wù)時(shí),使用靜態(tài)配置是不可行的,這個(gè)時(shí)候可以去注冊(cè)中心獲取可用的服務(wù)實(shí)例并調(diào)用。
Zookeeper
Zookeeper 是一個(gè)傳統(tǒng)的分布式協(xié)調(diào)服務(wù),它更多的被用來作為一個(gè)協(xié)調(diào)器使用,比如來協(xié)調(diào)管理 Hadoop 集群、協(xié)調(diào) Kafka 的 leader 選舉等。
為什么會(huì)有組件將其視為一個(gè)注冊(cè)中心使用?我想有幾個(gè)原因:
- Zookeeper 在分布式系統(tǒng)中具有更強(qiáng)的一致性和可靠性,可以確保各個(gè)服務(wù)的注冊(cè)信息保持一致。
- Zookeeper 使用內(nèi)存存儲(chǔ)數(shù)據(jù),具有很高的讀寫性能。這對(duì)于注冊(cè)中心來說非常關(guān)鍵,因?yàn)樗枰焖俚仨憫?yīng)客戶端的請(qǐng)求。
- Zookeeper 的 Watcher 機(jī)制可以讓客戶端監(jiān)聽指定節(jié)點(diǎn)的變化。當(dāng)某個(gè)節(jié)點(diǎn)(注冊(cè)中心)發(fā)生變化時(shí),Zookeeper 可以通知其他服務(wù)實(shí)現(xiàn)實(shí)時(shí)更新。
工作原理
以下圖為例,可以看到 Dubbo 是如何使用 Zookeeper 實(shí)現(xiàn)服務(wù)注冊(cè)/發(fā)現(xiàn)的。
- 服務(wù)提供者向 /dubbo/com.foo.BarService/providers 目錄下寫入自己的 URL 地址。
- 服務(wù)消費(fèi)者訂閱 /dubbo/com.foo.BarService/providers 目錄下的提供者 URL 地址。并向 /dubbo/com.foo.BarService/consumers 目錄下寫入自己的 URL 地址。
這里的目錄就是 Zookeeper 的數(shù)據(jù)結(jié)構(gòu),原理非常簡單,本質(zhì)上就是服務(wù)提供者和消費(fèi)者按照約定在 Zookeeper 上讀寫數(shù)據(jù),同時(shí)借用其 Watcher 機(jī)制、臨時(shí)節(jié)點(diǎn)和可靠性等特性高效的實(shí)現(xiàn)以下功能:
- 當(dāng)提供者服務(wù)出現(xiàn)斷電等異常停機(jī)時(shí),注冊(cè)中心能自動(dòng)刪除提供者信息。
- 當(dāng)注冊(cè)中心重啟時(shí),能自動(dòng)恢復(fù)注冊(cè)數(shù)據(jù)以及訂閱請(qǐng)求。
實(shí)現(xiàn)過程
注冊(cè)中心
下面通過 Zookeeper 的 Java API 實(shí)現(xiàn)一個(gè)只包含服務(wù)注冊(cè)/發(fā)現(xiàn)的注冊(cè)中心,代碼如下:
public class RegistrationCenter { // 連接信息 private String connectString = "192.168.10.11:2181,192.168.10.11:2182,192.168.10.11:2183"; // 超時(shí)時(shí)間 private int sessionTimeOut = 30000; private final String ROOT_PATH = "/servers"; private ZooKeeper client; public RegistrationCenter() { this(null); } public RegistrationCenter(Consumer<List<String>> consumer) { try { getConnection(null == consumer ? null : watchedEvent -> { //監(jiān)聽服務(wù)器地址的上下線 if (watchedEvent.getType() == Watcher.Event.EventType.NodeChildrenChanged) { try { consumer.accept(subServers()); } catch (Exception e) { e.printStackTrace(); } } }); Stat stat = client.exists(ROOT_PATH, false); if (stat == null) { //創(chuàng)建根節(jié)點(diǎn) client.create(ROOT_PATH, ROOT_PATH.getBytes(), ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT); } } catch (Exception e) { e.printStackTrace(); } } /** * @param serverName 將服務(wù)器注冊(cè)到zk集群時(shí),所需的服務(wù)名稱 * @param metadata 服務(wù)元數(shù)據(jù) * @throws Exception */ public void doRegister(String serverName, Metadata metadata) throws Exception { /** * ZooDefs.Ids.OPEN_ACL_UNSAFE: 此權(quán)限表示允許所有人訪問該節(jié)點(diǎn)(服務(wù)器) * CreateMode.EPHEMERAL_SEQUENTIAL: 由于服務(wù)器是動(dòng)態(tài)上下線的,上線后存在,下線后不存在,所以是臨時(shí)節(jié)點(diǎn) * 而服務(wù)器一般都是有序號(hào)的,所以是臨時(shí)、有序的節(jié)點(diǎn). */ String node = client.create(ROOT_PATH + "/" + serverName, metadata.toString().getBytes(), ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL_SEQUENTIAL); System.out.println(serverName + " 已經(jīng)上線"); } /** * 發(fā)現(xiàn)/訂閱服務(wù) */ public List<String> subServers() throws InterruptedException, KeeperException { List<String> zkChildren = client.getChildren(ROOT_PATH, true); List<String> servers = new ArrayList<>(); zkChildren.forEach(node -> { //拼接服務(wù)完整信息 try { byte[] data = client.getData(ROOT_PATH + "/" + node, false, null); servers.add(new String(data)); } catch (Exception e) { e.printStackTrace(); } }); return servers; } private void getConnection(Watcher watcher) throws IOException { this.client = new ZooKeeper(connectString, sessionTimeOut, watcher); } /** * 服務(wù)元數(shù)據(jù) */ public static class Metadata { public Metadata() { } public Metadata(String ip, int port) { this.ip = ip; this.port = port; } private String ip; private int port; public String getIp() { return ip; } public void setIp(String ip) { this.ip = ip; } public int getPort() { return port; } public void setPort(int port) { this.port = port; } @Override public String toString() { return "{" + "ip='" + ip + '\'' + ", port=" + port + '}'; } } }
該類中兩個(gè)核心方法:doRegister()
、subServers()
服務(wù)注冊(cè)和訂閱。
doRegister()
主要是往 Zookeeper 中創(chuàng)建了一個(gè)臨時(shí)節(jié)點(diǎn)數(shù)據(jù),臨時(shí)節(jié)點(diǎn)的優(yōu)勢(shì)就是當(dāng)服務(wù)出現(xiàn)斷電等異常停機(jī)時(shí),節(jié)點(diǎn)會(huì)自動(dòng)刪除。subServers()
則是去 Zookeeper 讀取了所有服務(wù)提供者的信息,并且監(jiān)聽了節(jié)點(diǎn)狀態(tài),當(dāng)節(jié)點(diǎn)發(fā)生創(chuàng)建、刪除、更新等事件時(shí)重新獲取服務(wù)者的信息,做到數(shù)據(jù)實(shí)時(shí)更新。
至此,一個(gè)簡單的注冊(cè)中心就完成了,當(dāng)然,如果要實(shí)現(xiàn)一個(gè)成熟的注冊(cè)中心,還要考慮負(fù)載均衡、高可用性和容錯(cuò)、服務(wù)治理和路由控制等功能,這里先不展開。
服務(wù)注冊(cè)
當(dāng)有了注冊(cè)中心,服務(wù)提供者就可以調(diào)用 doRegister()
進(jìn)行注冊(cè),代碼如下:
public class ProviderServer { public static void main(String[] args) throws Exception { RegistrationCenter registrationCenter = new RegistrationCenter(); registrationCenter.doRegister("provider", new RegistrationCenter.Metadata("127.0.0.1", 8080)); Thread.sleep(Long.MAX_VALUE); } }
服務(wù)發(fā)現(xiàn)
同樣,服務(wù)消費(fèi)者可以調(diào)用 subServers()
發(fā)現(xiàn)服務(wù)提供者,同時(shí)當(dāng)服務(wù)提供者發(fā)生變化時(shí)會(huì)通知到消費(fèi)者。代碼如下:
public class ConsumerServer { public static void main(String[] args) throws Exception { RegistrationCenter registrationCenter = new RegistrationCenter(newServers -> { System.out.println("服務(wù)更新了..."+newServers); }); List<String> servers = registrationCenter.subServers(); System.out.println(servers); Thread.sleep(Long.MAX_VALUE); } }
總結(jié)
服務(wù)注冊(cè)和服務(wù)發(fā)現(xiàn)功能是為了解決分布式系統(tǒng)中的服務(wù)管理和通信問題而設(shè)計(jì)的,經(jīng)過不斷的發(fā)展與負(fù)載均衡、健康監(jiān)測(cè)、服務(wù)治理和路由控制等功能完善成為一個(gè)注冊(cè)中心。服務(wù)注冊(cè)和服務(wù)發(fā)現(xiàn)有助于實(shí)現(xiàn)系統(tǒng)的彈性和可擴(kuò)展性,因?yàn)樾碌姆?wù)實(shí)例可以動(dòng)態(tài)地加入系統(tǒng),而無需手動(dòng)配置和修改已有的代碼。
以上就是基于Zookeeper實(shí)現(xiàn)服務(wù)注冊(cè)和服務(wù)發(fā)現(xiàn)功能的詳細(xì)內(nèi)容,更多關(guān)于Zookeeper服務(wù)注冊(cè)和發(fā)現(xiàn)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
knife4j3.0.3整合gateway和注冊(cè)中心的詳細(xì)過程
這篇文章主要介紹了knife4j3.0.3整合gateway和注冊(cè)中心的詳細(xì)過程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-03-03Spring超詳細(xì)講解事務(wù)和事務(wù)傳播機(jī)制
Spring事務(wù)的本質(zhì)就是對(duì)數(shù)據(jù)庫事務(wù)的支持,沒有數(shù)據(jù)庫事務(wù),Spring是無法提供事務(wù)功能的。Spring只提供統(tǒng)一的事務(wù)管理接口,具體實(shí)現(xiàn)都是由數(shù)據(jù)庫自己實(shí)現(xiàn)的,Spring會(huì)在事務(wù)開始時(shí),根據(jù)當(dāng)前設(shè)置的隔離級(jí)別,調(diào)整數(shù)據(jù)庫的隔離級(jí)別,由此保持一致2022-06-06Java生產(chǎn)者消費(fèi)者的三種實(shí)現(xiàn)方式
這篇文章主要介紹了Java生產(chǎn)者消費(fèi)者的三種實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-07-07@Async導(dǎo)致controller?404及失效原因解決分析
這篇文章主要為大家介紹了@Async導(dǎo)致controller?404失效問題解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07Springboot優(yōu)化內(nèi)置服務(wù)器Tomcat優(yōu)化方式(underTow)
本文詳細(xì)介紹了Spring Boot中Tomcat和Undertow服務(wù)器的配置和優(yōu)化,包括初始線程數(shù)、最大線程數(shù)、最小備用線程數(shù)、最大請(qǐng)求數(shù)等參數(shù)的優(yōu)化建議,以及在高并發(fā)場(chǎng)景下Undertow相對(duì)于Tomcat的優(yōu)勢(shì)2024-12-12Java實(shí)現(xiàn)簡易HashMap功能詳解
這篇文章主要介紹了Java實(shí)現(xiàn)簡易HashMap功能,結(jié)合實(shí)例形式詳細(xì)分析了Java實(shí)現(xiàn)HashMap功能相關(guān)原理、操作步驟與注意事項(xiàng),需要的朋友可以參考下2020-05-05idea導(dǎo)入工程時(shí)不能導(dǎo)入maven項(xiàng)目不能加入tomcatServer的原因
這篇文章主要介紹了idea導(dǎo)入工程時(shí)不能導(dǎo)入maven項(xiàng)目不能加入tomcatServer的原因及解決方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-09-09spring常用注解開發(fā)一個(gè)RESTful接口示例
這篇文章主要為大家介紹了使用spring常用注解開發(fā)一個(gè)RESTful接口示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2022-03-03