精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

靈感乍現!造了個與眾不同的Dubbo注冊中心擴展輪子

開發 前端
隨著這樣的習慣日積月累,低配的Mac上相繼跑了etcd、redis、mysql等等容器,重要的是還打開了N個IDEA窗口。

hello大家好呀,我是小樓。

作為一名基礎組件開發,服務好每一位業務開發同學是我們的義務(KPI)。

客服群里經常有業務開發同學丟來一段代碼、一個報錯,而我們,當然要微笑服務,耐心解答。

有的問題,憑借多年踩坑經驗,一眼就能看出,有的問題,看一眼代碼也能知道原因,但有的問題,還真就光憑看是看不出來的,這時,只能下載代碼,本地跑跑看了。

熟悉我的朋友都知道,我從事dubbo相關開(客)發(服)工作多年,所以我就來講一個dubbo問題排查過程中的有趣的事。

通常遇到看不能解決的問題時,先git拉取代碼,再導入IDEA,找到main方法點擊啟動,一頓操作下來,不出意外,肯定會有點小錯誤,比如這條:

Socket error occurred: localhost/127.0.0.1:2181: Connection refused

看到2181端口就知道這是本地沒有裝zookeeper(下文簡稱zk),問題不大,docker直接拉一個zk鏡像,起個容器就完事。

隨著這樣的習慣日積月累,低配的Mac上相繼跑了etcd、redis、mysql等等容器,重要的是還打開了N個IDEA窗口。

每當啟動一個新的項目時,風扇呼呼地直接將IDEA卡死。

這時,我陷入了思考,能不能少跑點程序?

etcd、redis、mysql暫時搞不定,但dubbo的注冊中心我熟啊!柿子當然要挑軟的捏。

需求梳理

在開干之前,得先梳理一下需求,于是我腦子閃現出無數個在本地測試時遇到的與dubbo注冊中心有關問題的瞬間,但仔細一捋,無外乎兩種:

  • 作為provider:最最最主要的就是不要阻斷應用啟動。
  • 作為consumer:

不要阻斷應用啟動。

可以發現并調用本地的provider。

可以調用遠程的provider。

可以手動指定調用任意provider。

除了這兩個功能上的需求,還得解決我們最初的問題:「不要依賴第三方服務」(如zk)。

調研

由于一開始就想到了利用dubbo注冊中心擴展來實現這個功能,為了不重復造輪子,翻了一下dubbo源碼,看看是否已經有相應的實現:

發現除了dubbo-registry-multicast之外都是依賴了第三方服務,所以這個multicast是啥呢?dubbo官方文檔說的很清楚:

乍一看很符合我們的需求,但仔細一想,還是有幾點不滿足:

  • 不一定能發現遠程的provider,如果大家代碼都是用的zk,而你把代碼拉下來注冊中心改成multicast是沒法發現遠程的服務的;
  • 沒法手動指定調用任意provider。

產品設計

服務發現得有個載體,要么通過第三方組件、要么通過網絡。但我們忽略了,在本地,磁盤也可以作為一個載體。

provider注冊向磁盤文件寫入,consumer訂閱即讀取磁盤文件,當磁盤文件有變更時通知consumer,大概是這么個樣子:

這樣設計有什么好處呢?

  • 不依賴其他服務,只是文件的讀寫,不會阻塞應用啟動。
  • consumer和provider都在本地時,可以像其他注冊中心(如zk、nacos等)一樣工作,對開發者完全透明。
  • 可以手動修改、指定調用任意provider。

唯一的缺點是,無法發現遠程的provider,但我們可以手動指定,也算是沒有大礙。

我們以dubbo 2.7.x版本的接口級服務發現來設計我們的產品,因為這個版本使用的最多。

首先要考慮的是如何去組織服務發現文件,由于是接口級服務發現,我們就按服務名來作為文件名,每個服務一個文件:

其次每個文件的內容怎么組織?最簡單的就是將dubbo注冊的URL直接寫入文件,每行一個URL,就像這樣:

但你可能發現了問題,這dubbo的URL有點長啊~如果讓我手動指定,豈不是很難做到?

這個問題好解決,我們實現一個簡寫版本的URL,比如有一行這樣簡寫,就將它還原為一個可用的URL。

127.0.0.1:20880

代碼實現

在實現之前首先要了解的是dubbo注冊中心擴展是如何編寫的,這塊直接看官方文檔:

https://dubbo.apache.org/zh/docs/v2.7/dev/impls/registry/

雖然我覺得看完了文檔你也不一定能實現一個dubbo注冊中心擴展,但別慌,先往下看,說不定看完了本文你也能自己寫一個。

先看一下代碼結構:

  • 項目命名為:dubbo-registry-mock,和dubbo源碼中的命名風格保持一致。
  • MockRegistry是注冊中心的核心實現。
  • MockRegistryFactory是mock registry的工廠,dubbo會通過這個類來創建MockRegistry。
  • org.apache.dubbo.registry.RegistryFactory這個文件是指定MockRegistryFactory該如何加載,即dubbo的SPI發現文件。

dubbo的注冊中心配置只需要改成:

dubbo.registry.address=mock://127.0.0.1:2181

這里起作用的只有mock,ip、port并不重要,只是占個位置。

當dubbo應用啟動時,讀取到配置的mock,會查找resources/META-INF.dubbo下的org.apache.dubbo.registry.RegistryFactory文件,這里它的內容為:

mock=org.newboo.MockRegistryFactory

于是去new出一個MockRegistryFactory。

注:newboo.org是我曾經注冊的一個域名,用來放博客,不過后來沒有續費,現在我的測試代碼中經常會出現這個包名。

MockRegistryFactory也很簡單,直接new一個MockRegistry:

public class MockRegistryFactory extends AbstractRegistryFactory {

@Override
protected Registry createRegistry(URL url) {
return new MockRegistry(url);
}
}

最后看核心的實現MockRegistry類:

public MockRegistry(URL url) {
super(url);
String basePath = DISCOVERY_DEFAULT_DIR;
if (StringUtils.isNotEmpty(url.getParameter(DISCOVERY_FILE_DIR_KEY))) {
basePath = url.getParameter(DISCOVERY_FILE_DIR_KEY);
}

mockService = new MockService(basePath);

ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1, new NamedThreadFactory("file_scan", true));
scheduledExecutorService.scheduleWithFixedDelay(new SubscribeScan(), 1000L, 5000, TimeUnit.MILLISECONDS);
}

這個構造方法,做了3件事情:

  • 獲取basePath,也就是服務發現的文件夾基礎路徑,有個默認值,也可以根據url的參數進行調整,如:
dubbo.registry.address=mock://127.0.0.1:2181?discovery_file=/tmp/mock-registry2
  • new一個MockService,承載了核心的服務發現邏輯,后面再說。
  • 啟動一個定時任務,每隔5秒去掃描一次文件,看文件是否有變化,如果有變化則通知consumer,詳細后面也會說。

MockRegistry繼承自FailbackRegistry,只需要實現它的doRegister、doUnregister、doSubscribe、doUnsubscribe、isAvailable幾個方法即可。

其中isAvailable是判斷注冊中心是否可用,我們直接返回true即可。

doUnsubscribe是取消訂閱,這里也啥都不用干,剩下3個方法我們將邏輯封裝在MockService:

@Override
public void doRegister(URL url) {
try {
mockService.writeUrl(url);
} catch (Throwable e) {
throw new RpcException("Failed to register " + url, e);
}
}

@Override
public void doUnregister(URL url) {
try {
mockService.removeUrl(url);
} catch (Throwable e) {
throw new RpcException("Failed to unregister " + url, e);
}
}

@Override
public void doSubscribe(URL url, NotifyListener listener) {
try {
List<URL> urls = mockService.getUrls(url.getServiceInterface());
listener.notify(urls);
} catch (ServiceNotChangeException ignored) {
} catch (Throwable e) {
throw new RpcException("Failed to subscribe " + url, e);
}
}

writeUrl直接獲取到文件名,往文件中append新的一行URL即可:

public void writeUrl(URL url) throws IOException {
String fileName = pathCenter.getServicePath(url.getServiceInterface());

// 寫入文件
String line = url.toFullString();
FileUtil.appendLine(fileName, line);
}

removeUrl先讀取文件,把要注銷的URL刪除,再把剩余內容覆蓋寫回文件即可:

public void removeUrl(URL url) throws IOException {
String fileName = pathCenter.getServicePath(url.getServiceInterface());
String line = url.toFullString();

List<String> lines = FileUtil.readLines(fileName);
lines = LinesUtil.removeLine(lines, line);

FileUtil.writeLines(fileName, lines);
}

getUrls去掃描文件,如果文件有變更,就把讀取到的最新的URL格式化后返回,之所以要格式化是因為可能會有簡寫的URL(見上文),文件是否有變更直接根據文件的最后更新時間來判斷,精確到毫秒,本地測試也夠用了:

 public List<URL> getUrls(String service) throws Exception {
if (!scan(service)) {
throw new ServiceNotChangeException();
}

String fileName = pathCenter.getServicePath(service);
List<String> lines = FileUtil.readLines(fileName);
List<URL> urls = new ArrayList<>(lines.size());
for (String line : lines) {
if (!LinesUtil.isSkipLine(line)) {
urls.add(format(line));
}
}
return urls;
}

其中scan如果返回false,說明文件沒有變更,直接忽略本次掃描。

最后一個SubscribeScan只需要把已經訂閱的接口拿出來,執行一次doSubscribe即可:

public class SubscribeScan implements Runnable {
@Override
public void run() {
try {
// 已經訂閱的url
Map<URL, Set<NotifyListener>> subscribeds = getSubscribed();
if (subscribeds == null || subscribeds.isEmpty()) {
return;
}

for (Map.Entry<URL, Set<NotifyListener>> entry : subscribeds.entrySet()) {
for (NotifyListener listener : entry.getValue()) {
doSubscribe(entry.getKey(), listener);
}
}
} catch (Throwable t) {
// ignore
}
}
}

看到這里可能有的同學問,為啥要輪詢,不用WatchService監聽文件的變更呢?我寫的時候也查了一下,并且debug了一下,發現WatchService的真實實現是PollingWatchService,而且它也是采用輪詢來實現的,不信可以打開這個類看看。

感覺和自己寫沒啥差別,所以我就自己寫了。

完整代碼已經上傳到了github:https://github.com/lkxiaolou/dubbo-registry-mock。

為了讓這個項目看起來更飽滿一點,還寫了一個README:

最后

如果你耐心看完了本文,且對dubbo有所了解,我相信你已經能自己寫一個dubbo注冊中心擴展。

如果你也經常在本地做測試,也可以用我寫的這個mock registry來試試,當然代碼和想法都有改進的地方,如果你有更好的想法也可以和我交流。

責任編輯:武曉燕 來源: 捉蟲大師
相關推薦

2022-02-10 20:09:24

Dubbo源碼Provider

2022-05-13 09:16:49

Python代碼

2015-08-06 10:14:15

造輪子facebook

2009-12-15 10:03:41

微軟數據中心服務器機柜

2011-09-19 13:57:47

Vista屏保注冊表

2020-09-08 08:45:39

jupyter插件代碼

2010-02-07 14:54:13

Android

2021-05-27 11:10:23

注冊中心業務

2023-02-06 17:27:48

2023-01-30 22:43:39

DubboZooKeeper

2025-06-26 01:25:00

2024-04-10 12:22:19

DubboNacos微服務

2022-09-14 08:22:50

AlloyDB高性能高可用性

2010-01-13 18:39:05

C++安裝

2023-06-25 14:35:27

網絡安全安全漏洞

2022-12-07 10:34:45

AST前端編譯

2021-03-09 16:38:48

加密貨幣比特幣貨幣

2009-12-17 16:53:13

.NET Framew

2022-01-05 11:40:36

Go特性語言

2010-06-17 14:42:31

RS-232C協議
點贊
收藏

51CTO技術棧公眾號

24小时成人在线视频| 二区在线视频| 中文精品视频| 亚洲视频在线观看| 日本不卡一区二区在线观看| 97超碰资源站在线观看| 成人av高清在线| 国产精品扒开腿做爽爽爽视频| 国内毛片毛片毛片毛片毛片| 高清日韩中文字幕| 日本韩国欧美国产| 看一级黄色录像| 毛片在线播放网站| 国产一区美女在线| 97超级碰在线看视频免费在线看| 中国1级黄色片| 国产 日韩 欧美 综合 一区| 欧美日韩电影一区| 乱人伦xxxx国语对白| 在线免费观看黄| 99久久精品99国产精品| 成人激情在线观看| 欧美日韩综合一区二区三区| 欧美 日韩 国产 一区| 亚洲图片欧美日产| 日本精品一二三区| 成人噜噜噜噜| 91福利资源站| 欧美爱爱视频免费看| av网站大全在线| 欧美韩日一区二区三区四区| 精品一卡二卡三卡四卡日本乱码 | 欧美日韩免费网站| 日韩欧美电影一区二区| 天天舔天天干天天操| 国产乱国产乱300精品| 国产成人精品网站| 精品成人av一区二区在线播放| 女生裸体视频一区二区三区| 日韩亚洲一区二区| 日韩免费成人av| 丝袜久久网站| 亚洲精品美女视频| 男人的天堂影院| 福利片一区二区| 精品捆绑美女sm三区| 五月天婷婷影视| 福利一区二区| 欧美视频一区二区三区四区 | 久久99久久99精品免视看婷婷 | 99精品在线免费观看| 欧洲生活片亚洲生活在线观看| 日韩精品一区二区三区久久| 国产污视频在线播放| 亚洲成av人片一区二区梦乃| 日韩一级片一区二区| 在线xxxx| 亚洲一区二区三区四区在线| 国产一级大片免费看| 中日韩高清电影网| 亚洲精品菠萝久久久久久久| 久久视频免费在线| 91麻豆免费在线视频| 有坂深雪av一区二区精品| 欧美精品在欧美一区二区| 污视频免费在线观看| 亚洲已满18点击进入久久| 免费网站在线观看视频| heyzo在线| 精品国产乱码久久久久久婷婷| 无罩大乳的熟妇正在播放| 成人影院av| 欧美性大战久久| 天堂av8在线| 日韩精品一区二区三区中文| 精品sm捆绑视频| 青青草视频成人| 成人无号精品一区二区三区| 久久亚洲综合国产精品99麻豆精品福利| 欧美性x x x| 亚洲国产电影| 国产精品91在线| 国产免费福利视频| 成人午夜免费视频| 久久亚裔精品欧美| 色欧美激情视频在线| 亚洲精品视频自拍| 免费毛片网站在线观看| 黄色成人免费网| 欧美日韩电影一区| 中国黄色片视频| 国产一区三区在线播放| 美日韩精品免费视频| 久久免费少妇高潮99精品| 久久久xxx| 91在线播放国产| 日本国产在线| 亚洲精品国产精华液| 国产第一页视频| 亚洲日日夜夜| 日韩激情av在线免费观看| 婷婷丁香综合网| 日韩图片一区| 91色在线观看| 日韩一区二区三区中文字幕| 亚洲三级免费电影| 欧美日韩亚洲一| 成人在线日韩| 亚洲欧洲免费视频| 欧美卡一卡二卡三| 久久精品午夜| 国产精品sss| 亚洲麻豆精品| 日本大香伊一区二区三区| 乳色吐息在线观看| av影片在线一区| 亚洲91精品在线| 亚洲天天综合网| 91一区二区在线观看| av磁力番号网| 久久久久久久性潮| 日韩黄在线观看| 久久亚洲AV无码| 韩国成人福利片在线播放| 免费在线观看91| 波多一区二区| 日韩一级高清毛片| 萌白酱视频在线| 日韩精品视频网| 久久久久久欧美精品色一二三四 | 亚洲乱码一区二区三区在线观看| 日本精品免费在线观看| 久久久免费毛片| 欧美激情视频网| 国产乱人乱偷精品视频a人人澡| 久久九九国产精品| 国产精品秘入口18禁麻豆免会员| 北条麻妃在线一区二区免费播放| 久久精品99久久香蕉国产色戒| 奴色虐av一区二区三区| 久久青草国产手机看片福利盒子 | 精品国产一区二区精华| 2021亚洲天堂| 国产精品99久久久久久宅男| eeuss中文| 国产精品一区三区在线观看| 久久在精品线影院精品国产| 国产美女自慰在线观看| 中文字幕亚洲精品在线观看| 五月天激情播播| 日韩精品免费一区二区三区| 国产精品天天狠天天看| √新版天堂资源在线资源| 在线观看日韩精品| 亚洲天堂岛国片| 久久国产精品区| 致1999电视剧免费观看策驰影院| 91精品国产色综合久久不卡粉嫩| 久久九九全国免费精品观看| 国产强被迫伦姧在线观看无码| 日韩毛片视频在线看| 欧美一级特黄aaa| 欧美精品成人| 国产一区二区黄色| 亚洲精品日产| 国产一区二区三区视频在线观看| 中文字幕精品在线观看| 国产精品电影院| 天堂网成人在线| 激情欧美日韩| 欧美lavv| 国产一区二区三区视频在线| 久久99热精品这里久久精品| 噜噜噜久久,亚洲精品国产品| 午夜成人免费电影| 人人妻人人藻人人爽欧美一区| 美女诱惑一区二区| 色哺乳xxxxhd奶水米仓惠香| 国产精品欧美大片| 国产精品久久久久免费a∨大胸| 美女av在线播放| 亚洲国产成人爱av在线播放| 无码人妻熟妇av又粗又大| 国产精品国产自产拍高清av王其| 亚洲制服在线观看| 99热精品在线观看| 先锋在线资源一区二区三区| 亚洲精品在线a| 日韩av手机在线看| 国产在线高潮| 亚洲裸体xxxx| av中文字幕播放| 色综合久久综合网欧美综合网| 免费黄色激情视频| 972aa.com艺术欧美| 色一情一区二区| 亚洲日本国产| 在线观看欧美一区| 欧美激情网址| 亚洲va欧美va在线观看| 在线观看网站免费入口在线观看国内 | 久久久蜜桃一区二区人| 亚洲精品一区国产精品| 91蜜桃臀久久一区二区| 国产精品久久av| 黄黄的网站在线观看| 日韩精品一区二区三区第95| 国产精品人人爽| 色婷婷一区二区| 久久久久久久久久网站| 久久久久久久久一| 国产女主播在线播放| 久久国产精品99精品国产| 草草久久久无码国产专区| 欧美激情第8页| 亚洲精品tv久久久久久久久| 亚洲ab电影| 91青青草免费观看| 日韩电影精品| 国产精品激情av电影在线观看| 9765激情中文在线| 欧美成人小视频| 婷婷在线视频| 国产亚洲综合久久| 污视频在线免费| 欧美电视剧在线看免费| 亚洲天堂中文字幕在线| 在线观看不卡视频| 亚洲欧美综合自拍| 午夜激情久久久| 国产一级片免费观看| 亚洲免费在线视频| 日韩亚洲欧美中文字幕| 欧美经典一区二区| 免费看污片网站| 久久婷婷国产综合精品青草| 在线免费观看a级片| 成人av第一页| a级片在线观看视频| 粉嫩一区二区三区性色av| 国产精品久久久久久久99| 九一九一国产精品| 久国产精品视频| 狠狠色狠狠色综合| 天天干天天色天天干| 极品少妇一区二区三区精品视频| 美女网站视频黄色| 美女视频一区二区| 在线观看免费视频高清游戏推荐| 日本亚洲视频在线| 视频二区在线播放| 另类人妖一区二区av| 老司机午夜性大片| 国产在线精品免费| 亚欧精品在线视频| 国产成人精品亚洲日本在线桃色| 国产精品99精品无码视亚| 国产成人自拍网| 欧美日韩一区二区三区四区五区六区| 成人精品免费网站| av网站有哪些| 国产片一区二区| 熟女av一区二区| 夜夜精品浪潮av一区二区三区| 久久精品无码人妻| 天天av天天翘天天综合网色鬼国产| 国产毛片aaa| 精品1区2区3区| 精品久久国产视频| 亚洲精品mp4| 成人三级黄色免费网站| 久久精品国产综合| 女同视频在线观看| 欧美中文字幕在线播放| 丁香婷婷久久| 99视频日韩| 欧美日韩123| 中文字幕精品在线播放| 夜夜嗨一区二区三区| 天天干天天干天天干天天干天天干| 国产一区视频在线看| 国产精品无码在线| 欧美激情中文不卡| 青青草手机在线观看| 一本色道a无线码一区v| 国产美女主播在线观看| 日韩av网站电影| 免费av在线网站| 97免费中文视频在线观看| 成人免费视频观看| 国产精品一区二区你懂得| 教室别恋欧美无删减版| 人妻无码一区二区三区四区| 久久综合中文| 不许穿内裤随时挨c调教h苏绵 | 日韩在线伦理| 国产日韩欧美黄色| 日韩福利视频一区| 一本二本三本亚洲码| 久久性色av| 成人在线观看一区二区| 欧美国产精品一区二区三区| 欧美被狂躁喷白浆精品| 欧美在线一二三四区| 黄色aaa毛片| www.日韩不卡电影av| 亚洲日本天堂| 成人国产一区二区| 爽成人777777婷婷| 99蜜桃臀久久久欧美精品网站| 国产成人午夜高潮毛片| 快灬快灬一下爽蜜桃在线观看| 亚洲国产一区二区三区| 91激情在线观看| 亚洲图中文字幕| 天堂av在线| 国产精品大全| 午夜久久99| 欧美一级小视频| 国产精品亲子乱子伦xxxx裸| av黄色在线播放| 亚洲精品白浆高清久久久久久| av色综合久久天堂av色综合在| 国产精品久久久久免费a∨| 日韩欧美影院| 可以在线看的av网站| 国产成人在线视频免费播放| 激情无码人妻又粗又大| 欧美性受xxxx| 成人性生交大片免费看午夜| 日本sm极度另类视频| 秋霞影视一区二区三区| 日韩中字在线观看| 成人午夜视频在线| 国产精品2020| 在线观看中文字幕av| 国产亚洲视频在线观看| 全球最大av网站久久| 欧美一级二级三级| 亚洲综合二区| 好吊日免费视频| 欧美天天综合色影久久精品| 天堂视频中文在线| 欧美一区二区三区免费观看| 欧美午夜18电影| 激情综合在线观看| 91麻豆6部合集magnet| 国产www在线| 亚洲日韩中文字幕在线播放| 91精品影视| 午夜精品一区二区三区四区| 免费在线观看精品| 欧美日韩国产黄色| 欧美日韩精品欧美日韩精品| 免费黄色电影在线观看| 91免费人成网站在线观看18| 欧美一区在线看| 日韩少妇一区二区| 亚洲成年人影院| 欧美女优在线| 国产精品第一第二| 久久国产亚洲精品| 日本一本在线视频| 亚洲国产日韩a在线播放性色| 天天操天天爱天天干| 欧美一级电影久久| 欧美综合另类| 亚洲欧美一区二区三区不卡| 亚洲一区二区在线免费观看视频 | 视频成人永久免费视频| 亚洲熟妇无码一区二区三区| 久久综合九色综合欧美就去吻| 日韩精选在线观看| 蜜月aⅴ免费一区二区三区| 国产精品中文字幕制服诱惑| 成人精品视频一区二区| 国产精品久久久久久久久免费丝袜 | 久久一级电影| 午夜激情视频网| 亚洲超碰97人人做人人爱| 九九在线视频| 国产欧美最新羞羞视频在线观看| 欧美 日韩 国产一区二区在线视频| 日本精品一二三区| 在线观看欧美黄色| 欧美午夜大胆人体| 欧美日韩亚洲一区二区三区在线观看 | zzijzzij亚洲日本少妇熟睡| 国产成人亚洲精品自产在线 | 蜜桃av噜噜一区二区三区小说| 91视频综合网| 亚洲欧美日韩直播| 日韩视频一二区| 日韩欧美黄色大片| 亚洲最色的网站| 中文字幕在线播放| 激情一区二区三区| 国产精品一区一区三区| 国产精品视频免费播放| 欧美成人在线网站|