詳解SpringCloud eureka服務(wù)狀態(tài)監(jiān)聽
一.前言
近期由于公司不同平臺項(xiàng)目之間的業(yè)務(wù)整合,需要做到相互訪問! 每個(gè)平臺均有自己的注冊中心和服務(wù),且注冊中心相互之間并沒有相互注冊!
借助spring的事件監(jiān)聽,在eureka-server端監(jiān)聽服務(wù)注冊,將所有服務(wù)的ip和port存放至redis庫,然后讓其他平臺服務(wù)通過redis庫獲取ip和端口號,進(jìn)而進(jìn)行http調(diào)用.結(jié)構(gòu)圖如下:

二.事件解析
事件列表
在org.springframework.cloud.netflix.eureka.server.event包下會發(fā)現(xiàn)如下類:

EurekaInstanceCanceledEvent: 服務(wù)下線事件EurekaInstanceRegisteredEvent: 服務(wù)注冊事件EurekaInstanceRenewedEvent: 服務(wù)續(xù)約事件EurekaRegistryAvailableEvent: eureka注冊中心啟動事件EurekaServerStartedEvent: eureka server啟動時(shí)間
源碼分析
打開org.springframework.cloud.netflix.eureka.server.InstanceRegistry類,會發(fā)現(xiàn)當(dāng)eureka服務(wù)續(xù)約、注冊、取消等時(shí),spring會publish不同的事件,對應(yīng)的事件類就是上面的列表.
續(xù)約事件
@Override
public boolean renew(final String appName, final String serverId,
boolean isReplication) {
log("renew " + appName + " serverId " + serverId + ", isReplication {}"
+ isReplication);
List<Application> applications = getSortedApplications();
for (Application input : applications) {
if (input.getName().equals(appName)) {
InstanceInfo instance = null;
for (InstanceInfo info : input.getInstances()) {
if (info.getId().equals(serverId)) {
instance = info;
break;
}
}
// 發(fā)布續(xù)約事件
publishEvent(new EurekaInstanceRenewedEvent(this, appName, serverId,
instance, isReplication));
break;
}
}
return super.renew(appName, serverId, isReplication);
}
注冊事件
@Override
public void register(InstanceInfo info, int leaseDuration, boolean isReplication) {
handleRegistration(info, leaseDuration, isReplication);
super.register(info, leaseDuration, isReplication);
}
private void handleRegistration(InstanceInfo info, int leaseDuration,
boolean isReplication) {
log("register " + info.getAppName() + ", vip " + info.getVIPAddress()
+ ", leaseDuration " + leaseDuration + ", isReplication "
+ isReplication);
// 發(fā)布注冊事件
publishEvent(new EurekaInstanceRegisteredEvent(this, info, leaseDuration,
isReplication));
}
事件監(jiān)聽
通過上面的源碼追溯,我們已經(jīng)得到對應(yīng)的事件類了,所以現(xiàn)在要做的僅僅是監(jiān)聽對應(yīng)的事件即可,至此已經(jīng)完成了我們所需要對事件監(jiān)聽后的業(yè)務(wù)處理!
@Component
public class EurekaStateChangeListener {
@Value("${iptable.platform}")
private String platform;
@Autowired
private RedisTemplate<String, String> redisTemplate;
private static Logger logger = LoggerFactory.getLogger(EurekaStateChangeListener.class);
private static final String COLON = ":";
@EventListener//(condition = "#event.replication==false")
public void listen(EurekaInstanceCanceledEvent eurekaInstanceCanceledEvent) {
// 服務(wù)斷線事件
String appName = eurekaInstanceCanceledEvent.getAppName();
String serverId = eurekaInstanceCanceledEvent.getServerId();
Objects.requireNonNull(appName, "服務(wù)名不能為空!");
SetOperations<String, String> opsForSet = redisTemplate.opsForSet();
opsForSet.remove((platform + appName).toLowerCase(), serverId);
logger.info(">>>>>>> 失效服務(wù):{},已被剔除!", serverId);
}
@EventListener//(condition = "#event.replication==false")
public void listen(EurekaInstanceRegisteredEvent event) {
// 服務(wù)注冊
InstanceInfo instanceInfo = event.getInstanceInfo();
String appName = instanceInfo.getAppName();
Objects.requireNonNull(appName, "服務(wù)名不能為空!");
SetOperations<String, String> opsForSet = redisTemplate.opsForSet();
opsForSet.add((platform + appName).toLowerCase(), instanceInfo.getIPAddr() + COLON + instanceInfo.getPort());
logger.info(">>>>>>> 服務(wù)名:{},端口號:{},已緩存至redis", appName, instanceInfo.getPort());
}
@EventListener//(condition = "#event.replication==false")
public void listen(EurekaInstanceRenewedEvent event) {
// 服務(wù)續(xù)約
logger.info(">>>>>>>>>>>>>>>Server續(xù)約:" + event.getServerId());
}
@EventListener
public void listen(EurekaRegistryAvailableEvent event) {
// 注冊中心啟動
logger.info(">>>>>>>>>>>>>>>Server注冊中心:" + event);
}
@EventListener
public void listen(EurekaServerStartedEvent event) {
// Server啟動
logger.info(">>>>>>>>>>>>>>>Server啟動:" + event);
}
}
注意事項(xiàng)
[ ] 版本問題:
當(dāng)時(shí)項(xiàng)目組用的SpringCloud版本是Brixton.RELEASE,該版本有一個(gè)問題就是服務(wù)注冊和下線并不會出發(fā)對應(yīng)的事件,所以導(dǎo)致一直監(jiān)聽不到.解決的辦法也很簡單,只要升級版本即可,我已經(jīng)升級到最新版本Finchley.RELEASE.
傳送門,點(diǎn)我
[ ] 重復(fù)監(jiān)聽:
例如,在續(xù)約的時(shí)候,eureka會發(fā)出2條EurekaInstanceRenewedEvent事件,但是2條事件的屬性卻不一樣!一個(gè)事件的屬性replication為true,另外一個(gè)為false.如果我們只想處理replication=true的事件,如下配置即可:
@EventListener(condition = "#event.replication==false")
public void listen(EurekaInstanceRenewedEvent event) {
// 服務(wù)續(xù)約
logger.info(">>>>>>>>>>>>>>>Server續(xù)約:" + event.getServerId());
}
GitHub代碼,點(diǎn)我
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Quarkus集成open api接口使用swagger ui展示
這篇文章主要為大家介紹了Quarkus集成open?api接口使用swagger?ui的展示示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2022-02-02
SpringBoot使用MockMvc進(jìn)行Web集成測試的示例詳解
MockMvc?是一個(gè)測試框架,可以模擬?HTTP?請求和響應(yīng),在本文中,我們將介紹如何使用MockMvc進(jìn)行Web集成測試,以及如何編寫測試用例來測試Spring?MVC控制器,希望對大家有所幫助2023-06-06
springboot中JSONObject遍歷并替換部分json值
這篇文章主要介紹了springboot中JSONObject遍歷并替換部分json值,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11
java設(shè)計(jì)模式之裝飾模式詳細(xì)介紹
這篇文章主要介紹了java設(shè)計(jì)模式之裝飾模式,有需要的朋友可以參考一下2013-12-12
SpringBoot攔截器實(shí)現(xiàn)登錄攔截的示例代碼
本文主要介紹了SpringBoot攔截器實(shí)現(xiàn)登錄攔截,文中根據(jù)實(shí)例編碼詳細(xì)介紹的十分詳盡,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03
ByteArrayOutputStream簡介和使用_動力節(jié)點(diǎn)Java學(xué)院整理
ByteArrayOutputStream 是字節(jié)數(shù)組輸出流。它繼承于OutputStream。這篇文章主要介紹了ByteArrayOutputStream簡介和使用,需要的朋友可以參考下2017-05-05

