springboot 注冊(cè)服務(wù)注冊(cè)中心(zk)的兩種方式詳解
在使用springboot進(jìn)行開(kāi)發(fā)的過(guò)程中,我們經(jīng)常需要處理這樣的場(chǎng)景:在服務(wù)啟動(dòng)的時(shí)候,需要向服務(wù)注冊(cè)中心(例如zk)注冊(cè)服務(wù)狀態(tài),以便當(dāng)服務(wù)狀態(tài)改變的時(shí)候,可以故障摘除和負(fù)載均衡。
我遇到過(guò)兩種注冊(cè)的途徑:
1、在Spring的webapplication啟動(dòng)完成后,直接進(jìn)行注冊(cè);
2、在servlet容器啟動(dòng)完成后,通過(guò)listener進(jìn)行注冊(cè)。
本文通過(guò)一個(gè)demo講述一下這兩種注冊(cè)方式,使用的是傳統(tǒng)的向zk注冊(cè)的方案。
1、Spring webapplication啟動(dòng)完成后注冊(cè)
先上代碼看一下
@SpringBootApplication
public class WebApplication {
private static final Logger logger = LoggerFactory.getLogger(WebApplication.class);
private static volatile boolean IS_REGISTRY = false;
public static void main(String[] args) {
ApplicationContext context = run(WebApplication.class, args);
if (IS_REGISTRY) {
logger.info("注冊(cè)2: WebApplication啟動(dòng)完成后");
ZkClient zkClient = context.getBean(ZkClient.class);
zkClient.register();
IS_REGISTRY = true;
logger.info("注冊(cè)2: 注冊(cè)成功");
}
}
}
這里,我們?cè)赪ebApplication中,獲取zkClient,并進(jìn)行注冊(cè)。
這里需要說(shuō)明一點(diǎn),我們這里通過(guò)ApplicationContext來(lái)獲取zkClient的bean,原因是在webApplication的初始化過(guò)程中你不能用Autowired的方式注入Bean,因?yàn)樵趙ebApplication啟動(dòng)過(guò)程中才會(huì)讀所有的configuration并將bean初始化完成,在沒(méi)有完成初始化之前,你不能注入bean。
關(guān)于注冊(cè)的詳細(xì)代碼這里不展開(kāi)了。
2、在servlet容器初始化完成后,通過(guò)listener的方式進(jìn)行注冊(cè)
照樣先上代碼
@WebListener
public class RegisterListener implements ServletContextListener {
protected final Logger logger = LoggerFactory.getLogger(this.getClass());
private static volatile boolean IS_REGISTRY = false;
@Autowired
private ZkClient zkClient;
@Override
public void contextInitialized(ServletContextEvent servletContextEvent) {
try {
if (!IS_REGISTRY) {
logger.info("注冊(cè)1: Servelet容器啟動(dòng)成功后");
zkClient.register();
logger.info("注冊(cè)1: 注冊(cè)成功");
}
IS_REGISTRY = true;
} catch (Exception e) {
IS_REGISTRY = false;
logger.info("注冊(cè)1: 注冊(cè)失敗");
}
}
@Override
public void contextDestroyed(ServletContextEvent servletContextEvent) {
if (IS_REGISTRY) {
zkClient.stop();
}
}
}
你需要先寫(xiě)一個(gè)listener,這個(gè)listener實(shí)現(xiàn)ServletContextListener接口,并且用@WebListener進(jìn)行注解,這是springboot注解式的listener書(shū)寫(xiě)方式。
在servlet容器啟動(dòng)成功之后,會(huì)調(diào)用這個(gè)監(jiān)聽(tīng)器的contextInitialized方法,servlet容器如果一旦銷(xiāo)毀,不能提供服務(wù)了,會(huì)調(diào)用監(jiān)聽(tīng)器的contextDestroyed方法。換句話(huà)說(shuō),這個(gè)監(jiān)聽(tīng)器在監(jiān)聽(tīng)servlet容器的狀態(tài)。
然后你只需要在application主類(lèi)中打開(kāi)listener配置就好。
@ServletComponentScan
@SpringBootApplication
public class WebApplication {
}
3、這兩種方式的比較
對(duì)于一個(gè)對(duì)外提供http協(xié)議的web服務(wù),在語(yǔ)義上servlet容器的注冊(cè)會(huì)顯得清晰一些,但是如果你的spring容器啟動(dòng)時(shí)間過(guò)長(zhǎng)的話(huà),可能出現(xiàn)servlet初始化完成,并且已經(jīng)注冊(cè),但是服務(wù)不能對(duì)外提供訪(fǎng)問(wèn)的gap time,所以我一般還是使用第一種方式進(jìn)行注冊(cè)。
這種場(chǎng)景是這樣的

可以看到,當(dāng)servlet注冊(cè)成功之后,其實(shí)webapplication還沒(méi)有啟動(dòng)完成,這個(gè)時(shí)候服務(wù)是不能正常提供訪(fǎng)問(wèn)的。

在zk上可以看到,兩次注冊(cè)都已經(jīng)成功了。
總結(jié)
以上所述是小編給大家介紹的springboot 注冊(cè)服務(wù)注冊(cè)中心(zk)的兩種方式詳解,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
- 詳解Spring Cloud Consul 實(shí)現(xiàn)服務(wù)注冊(cè)和發(fā)現(xiàn)
- spring cloud consul注冊(cè)的服務(wù)報(bào)錯(cuò)critical的解決
- spring cloud consul使用ip注冊(cè)服務(wù)的方法示例
- springcloud使用consul作為配置中心
- 詳解使用Spring Cloud Consul實(shí)現(xiàn)服務(wù)的注冊(cè)和發(fā)現(xiàn)
- SpringBoot + Spring Cloud Consul 服務(wù)注冊(cè)和發(fā)現(xiàn)詳細(xì)解析
相關(guān)文章
java string類(lèi)型轉(zhuǎn)換boolean類(lèi)型的方法
下面小編就為大家?guī)?lái)一篇java string類(lèi)型轉(zhuǎn)換boolean類(lèi)型的方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-11-11
IntelliJ IDEA同步代碼時(shí)版本沖突而產(chǎn)生出的incoming partial文件問(wèn)題的解決辦法
今天小編就為大家分享一篇關(guān)于IntelliJ IDEA同步代碼時(shí)版本沖突而產(chǎn)生出的incoming partial文件問(wèn)題的解決辦法,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2018-10-10
Java實(shí)現(xiàn)在正則表達(dá)式中控制大小寫(xiě)的方法
這篇文章主要介紹了Java實(shí)現(xiàn)在正則表達(dá)式中控制大小寫(xiě)的方法,結(jié)合實(shí)例形式分析了java正則表達(dá)式中傳遞控制參數(shù)的功能與相關(guān)操作技巧,需要的朋友可以參考下2017-04-04

