SpringBoot實(shí)現(xiàn)服務(wù)接入nacos注冊(cè)中心流程詳解
概述
某些場(chǎng)景下只需要把springboot微服務(wù)化而不想引入springcloud如何實(shí)現(xiàn)的呢?
下面我們介紹nacos注冊(cè)中心方案。
接入nacos注冊(cè)中心
springboot服務(wù)pom文件
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>com.study</groupId>
<artifactId>practice</artifactId>
<version>1.0.0-SNAPSHOT</version>
</parent>
<artifactId>practice-demo</artifactId>
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.1.3.RELEASE</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<java.version>1.8</java.version>
</properties>
<!-- springboot -->
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
<exclusions>
<exclusion>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-logging</artifactId>
</exclusion>
</exclusions>
</dependency>
<!--test -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-log4j2</artifactId>
</dependency>
<!--nacos注冊(cè)中心-->
<dependency>
<groupId>com.alibaba.boot</groupId>
<artifactId>nacos-discovery-spring-boot-starter</artifactId>
<version>0.2.7</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<version>2.1.3.RELEASE</version>
<executions>
<execution>
<goals>
<goal>repackage</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
</project>application.properties配置
# nacos注冊(cè)
spring.application.name=file-server-service
nacos.discovery.server-addr=192.168.1.1:8848,192.168.1.2:8848
nacos.discovery.auto-register=true
nacos.discovery.enabled=true
# 指定微服務(wù)注冊(cè)那個(gè)地址
nacos.discovery.register.ip=
源碼分析
主流程:
1、nacos-discovery-spring-boot-starter 啟動(dòng)服務(wù)通過(guò)SPI機(jī)制掃描到nacos-discovery-spring-boot-autoconfigure包。

2、nacos-discovery-spring-boot-autoconfigure項(xiàng)目通過(guò)自動(dòng)裝配功能裝配nacos客戶端

3、 Nacos自動(dòng)配置服務(wù)實(shí)現(xiàn)Spring的應(yīng)用監(jiān)聽器用來(lái)注冊(cè)nacos服務(wù)
NacosDiscoveryAutoRegister implements ApplicationListener
4、NacosDiscoveryAutoRegister監(jiān)聽到spring的ServletWebServerInitializedEvent事件后把springboot服務(wù)注冊(cè)到nacos注冊(cè)中心
5、調(diào)用nacos-client jar包中的com.alibaba.nacos.client.naming.net.NamingProxy#registerService完成服務(wù)注冊(cè)
核心處理邏輯:com.alibaba.nacos.client.naming.net.NamingProxy.java
public String reqAPI(String api, Map<String, String> params, String body, List<String> servers, String method) throws NacosException {
params.put("namespaceId", this.getNamespaceId());
if (CollectionUtils.isEmpty(servers) && StringUtils.isEmpty(this.nacosDomain)) {
throw new NacosException(400, "no server available");
} else {
NacosException exception = new NacosException();
// 如果nacos.discovery.server-addr是逗號(hào)分隔的列表走改分組
if (servers != null && !servers.isEmpty()) {
Random random = new Random(System.currentTimeMillis());
int index = random.nextInt(servers.size());
int i = 0;
while(i < servers.size()) {
String server = (String)servers.get(index);
try {
return this.callServer(api, params, body, server, method);
} catch (NacosException var13) {
exception = var13;
if (LogUtils.NAMING_LOGGER.isDebugEnabled()) {
LogUtils.NAMING_LOGGER.debug("request {} failed.", server, var13);
}
index = (index + 1) % servers.size();
++i;
}
}
}
// 如果nacos.discovery.server-addr只配置一個(gè)服務(wù)地址而不是逗號(hào)分隔的多個(gè)服務(wù)地址
if (StringUtils.isNotBlank(this.nacosDomain)) {
int i = 0;
while(i < 3) {
try {
return this.callServer(api, params, body, this.nacosDomain, method);
} catch (NacosException var12) {
exception = var12;
if (LogUtils.NAMING_LOGGER.isDebugEnabled()) {
LogUtils.NAMING_LOGGER.debug("request {} failed.", this.nacosDomain, var12);
}
++i;
}
}
}
...
}
}
/**
調(diào)用nacos服務(wù)器,把springboot服務(wù)注冊(cè)為微服務(wù)
使用服務(wù)注冊(cè)接口:http://xxx:xxx/nacos/v1/ns/instance
**/
public String callServer(String api, Map<String, String> params, String body, String curServer, String method) throws NacosException {
long start = System.currentTimeMillis();
long end = 0L;
this.injectSecurityInfo(params);
List<String> headers = this.builderHeaders();
String url;
// 如果nacos.discovery.server-addr地址不是http://或者h(yuǎn)ttps://開頭走該分組
if (!curServer.startsWith("https://") && !curServer.startsWith("http://")) {
// 如果只寫了ip地址,會(huì)追加默認(rèn)的8848端口
if (!curServer.contains(":")) {
curServer = curServer + ":" + this.serverPort;
}
// 添加http前綴,http或者h(yuǎn)ttps,配置項(xiàng)com.alibaba.nacos.client.naming.tls.enable確定是http還是https
url = HttpClient.getPrefix() + curServer + api;
} else {
url = curServer + api;
}
HttpResult result = HttpClient.request(url, headers, params, body, "UTF-8", method);
end = System.currentTimeMillis();
MetricsMonitor.getNamingRequestMonitor(method, url, String.valueOf(result.code)).observe((double)(end - start));
if (200 == result.code) {
return result.content;
} else if (304 == result.code) {
return "";
} else {
throw new NacosException(result.code, result.content);
}
}小結(jié)
nacos.discovery.server-addr配置項(xiàng)支持的方式:
單個(gè)ip:port形式
nacos.discovery.server-addr=192.168.10.18:8858
多個(gè)ip:port形式
nacos.discovery.server-addr=192.168.10.18:8858,192.168.10.19:8858
域名方式(http://或者h(yuǎn)ttps://開頭)
nacos.discovery.server-addr=https://www.xxx
從源碼我們可以看出,其實(shí)nacos.discovery.server-addr配置多個(gè)地址,nacos會(huì)隨機(jī)選擇一個(gè)服務(wù)器地址,如果注冊(cè)成功就返回了,不會(huì)去處理其他的服務(wù)地址,除非一個(gè)地址注冊(cè)失敗才會(huì)使用其他的地址注冊(cè);nacos集群情況下,最好配置多個(gè)地址,放在一個(gè)nacos注冊(cè)失敗導(dǎo)致服務(wù)注冊(cè)不上的問(wèn)題。
源碼流程圖

到此這篇關(guān)于SpringBoot實(shí)現(xiàn)服務(wù)接入nacos注冊(cè)中心流程詳解的文章就介紹到這了,更多相關(guān)SpringBoot接入nacos內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Springboot+Mybatis中typeAliasesPackage正則掃描實(shí)現(xiàn)方式
這篇文章主要介紹了Springboot+Mybatis中typeAliasesPackage正則掃描實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
Idea調(diào)用WebService的關(guān)鍵步驟和注意事項(xiàng)
這篇文章主要介紹了如何在Idea中調(diào)用WebService,包括理解WebService的基本概念、獲取WSDL文件、閱讀和理解WSDL文件、選擇對(duì)接測(cè)試工具或方式、發(fā)送請(qǐng)求和接收響應(yīng)、處理響應(yīng)結(jié)果以及錯(cuò)誤處理,需要的朋友可以參考下2025-01-01
Java使用Arrays.asList報(bào)UnsupportedOperationException的解決
這篇文章主要介紹了Java使用Arrays.asList報(bào)UnsupportedOperationException的解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04
springboot中application.yml多環(huán)境生效規(guī)則說(shuō)明
這篇文章主要介紹了springboot中application.yml多環(huán)境生效規(guī)則說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-07-07
SpringMVC基于注解方式實(shí)現(xiàn)上傳下載
本文主要介紹了SpringMVC基于注解方式實(shí)現(xiàn)上傳下載,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04
springboot+jwt+微信小程序授權(quán)登錄獲取token的方法實(shí)例
本文主要介紹了springboot+jwt+微信小程序授權(quán)登錄獲取token的方法實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03

