Nacos服务发现原理分析
微服务将自己的实例注册到
nacos
注册中心,nacos服务端存储了注册列表,然后通过ribbon
调用服务,具体是如何调用?如果nacos
服务挂了,还能正常调用服务吗?调用的服务列表发生变化,调用方是如何感知变化的?带着这些问题,来探索一下服务发现的原理。
版本 2.1.1
Nacos Server:2.1.1
spring-cloud-starter-alibaba:2.1.1.RELEASE
spring-boot:2.1.1.RELEASE
spring-cloud-starter-netflix-ribbon:2.1.1.RELEASE
客户端和服务端版本号都为
2.1.1
。
从 Ribbon 讲起
使用ribbon
来调用服务,就添加ribbon
依赖:
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-ribbon</artifactId>
</dependency>
ribbon
依赖包含spring-cloud-commons
依赖,而在spring-cloud-commons
包中spring.factories
自动配置LoadBalancerAutoConfiguration
类:
@LoadBalanced
@Autowired(required = false)
private List<RestTemplate> restTemplates = Collections.emptyList();
@Bean
public LoadBalancerInterceptor ribbonInterceptor(
LoadBalancerClient loadBalancerClient,
LoadBalancerRequestFactory requestFactory) {
return new LoadBalancerInterceptor(loadBalancerClient, requestFactory);
}
只要标注了
@LoadBalanced
注解的restTemplates
都会添加负载均衡拦截器LoadBalancerInterceptor
。
使用Ribbon
组件调用服务:
restTemplate.getForObject("http://service-name",String.class);
restTemplate
的http
请求方法,最终会调用到doExecute
方法。doExecute
在发起http
请求之前,会先执行LoadBalancerInterceptor
负载均衡拦截器的intercept
方法。 该方法调用execute
方法。
而在execute
方法中,主要有两个方法:
ILoadBalancer loadBalancer = getLoadBalancer(serviceId);
Server server = getServer(loadBalancer, hint);
execute
先通过getLoadBalancer
获取ILoadBalancer
实例,然后再通过getServer
获取Server
实例。
getLoadBalancer
最终会调用Ribbon
的ServerList
接口,具体调用流程:
getLoadBalancer() ->
ZoneAwareLoadBalancer ->
DynamicServerListLoadBalancer ->
restOfInit()->
updateListOfServers()->
ServerList.getUpdatedListOfServers()->
Nacos
实现类NacosServerList
实现了ServerList
接口。
总之我们在进行微服务调用的时候,
Ribbon
最终会调用NacosServerList
类中的getUpdatedListOfServers
方法。
Nacos 获取服务
NacosServerList
类的getUpdatedListOfServers
方法调用了该类的getServers
方法:
private List<NacosServer> getServers() {
try {
// 获取分组
String group = discoveryProperties.getGroup();
// 重点,查询实例列表
List<Instance> instances = discoveryProperties.namingServiceInstance()
.selectInstances(serviceId, group, true);
return instancesToServerList(instances);
}
catch (Exception e) {
throw new IllegalStateException(
"Can not get service instances from nacos, serviceId=" + serviceId,
e);
}
}
重点看NacosNamingService
类的selectInstances
方法,会调用以下selectInstances
三个重载方法:
@Override
public List<Instance> selectInstances(String serviceName, String groupName, boolean healthy) throws NacosException {
return selectInstances(serviceName, groupName, healthy, true);
}
@Override
public List<Instance> selectInstances(String serviceName, String groupName, boolean healthy, boolean subscribe) throws NacosException {
return selectInstances(serviceName, groupName, new ArrayList<String>(), healthy, subscribe);
}
@Override
public List<Instance> selectInstances(String serviceName, String groupName, List<String> clusters, boolean healthy, boolean subscribe) throws NacosException {
ServiceInfo serviceInfo;
// 默认订阅
if (subscribe) {
// 获取服务,这是重点
serviceInfo = hostReactor.getServiceInfo(NamingUtils.getGroupedName(serviceName, groupName), StringUtils.join(clusters, ","));
} else {
serviceInfo = hostReactor.getServiceInfoDirectlyFromServer(NamingUtils.getGroupedName(serviceName, groupName), StringUtils.join(clusters, ","));
}
return selectInstances(serviceInfo, healthy);
}
最后一个selectInstances
方法里面的hostReactor.getServiceInfo
方法是获取服务的核心方法:
public ServiceInfo getServiceInfo(final String serviceName, final String clusters) {
NAMING_LOGGER.debug("failover-mode: " + failoverReactor.isFailoverSwitch());
String key = ServiceInfo.getKey(serviceName, clusters);
if (failoverReactor.isFailoverSwitch()) {
return failoverReactor.getService(key);
}
// 先在本地缓存查询
ServiceInfo serviceObj = getServiceInfo0(serviceName, clusters);
// 查询不到
if (null == serviceObj) {
serviceObj = new ServiceInfo(serviceName, clusters);
serviceInfoMap.put(serviceObj.getKey(), serviceObj);
updatingMap.put(serviceName, new Object());
// 请求Nacos Server实例,并更新服务实例
updateServiceNow(serviceName, clusters);
updatingMap.remove(serviceName);
} else if (updatingMap.containsKey(serviceName)) {
if (UPDATE_HOLD_INTERVAL > 0) {
// hold a moment waiting for update finish
synchronized (serviceObj) {
try {
serviceObj.wait(UPDATE_HOLD_INTERVAL);
} catch (InterruptedException e) {
NAMING_LOGGER.error("[getServiceInfo] serviceName:" + serviceName + ", clusters:" + clusters, e);
}
}
}
}
// 定时更新本地缓存
scheduleUpdateIfAbsent(serviceName, clusters);
return serviceInfoMap.get(serviceObj.getKey());
}
getServiceInfo
是服务发现的核心方法,先查询serviceInfoMap
集合中查询本地缓存,本地缓存查询不到就请求Nacos Server
实例,并更新本地缓存。
请求Nacos Server
实例,实际就是发送http
请求Nacos Server
:
public void updateServiceNow(String serviceName, String clusters) {
ServiceInfo oldService = getServiceInfo0(serviceName, clusters);
try {
// 调用 Nacos Server 查询服务
String result = serverProxy.queryList(serviceName, clusters, pushReceiver.getUDPPort(), false);
// 结果不为空,更新缓存
if (StringUtils.isNotEmpty(result)) {
processServiceJSON(result);
}
} catch (Exception e) {
NAMING_LOGGER.error("[NA] failed to update serviceName: " + serviceName, e);
} finally {
if (oldService != null) {
synchronized (oldService) {
oldService.notifyAll();
}
}
}
}
//向 Nacos Server发起 HTTP 列表查询
public String queryList(String serviceName, String clusters, int udpPort, boolean healthyOnly)throws NacosException {
final Map<String, String> params = new HashMap<String, String>(8);
params.put(CommonParams.NAMESPACE_ID, namespaceId);
params.put(CommonParams.SERVICE_NAME, serviceName);
params.put("clusters", clusters);
params.put("udpPort", String.valueOf(udpPort));
params.put("clientIP", NetUtils.localIP());
params.put("healthyOnly", String.valueOf(healthyOnly));
return reqAPI(UtilAndComs.NACOS_URL_BASE + "/instance/list", params, HttpMethod.GET);
}
queryList
方法主要封装号请求参数,然后向Nacos Server
服务端发送http
请求。
当服务端实例发生改变时,Nacos Server
会推送最新的实例给服务端。
服务发现是先获取本地缓存,如果没有本地缓存,就请求Nacos Server
服务端获取数据,如果Nacos Server
挂了,也不会影响服务的调用。
总结
Ribbon
- 项目启动时,会创建一个负载均衡拦截器。
- 从
Ribbon
发起服务请求开始,最终会调用到拦截器的拦截方法。 - 拦截方法又调用
ServerList
获取实例接口,而NacosServerList
实现获取实例列表。
Nacos
调用服务NacosServerList
实现了获取服务实例列表。NacosServerList
类selectInstances
方法最终调用了hostReactor.getServiceInfo
方法getServiceInfo
方法先从serviceInfoMap
集合中获取本地缓存,如果本地缓存找不到,就请求Nacos Server
获取服务实例,并更新本地缓存。- 获取服务之后,定时更新本地缓存。
参考
最新文章
- asp.net mvc 5 web api 关于Requested resource does not support options 问题
- SDDC-SDN-SDS
- 关于未捕获异常的处理(WPF)
- Hibernate 配置 双向 对多关联 (未完待续&#183;&#183;&#183;&#183;&#183;&#183;&#183;)
- VS中Debug和Realease、及静态库和动态库的区别整理
- openfire+asmack搭建的安卓即时通讯(一) 15.4.7
- smarty简单介绍
- vlc分析
- mongodb 的js脚本或pymongodb脚本修改数据库的字段值
- 用Unitils测试BaseDao遇到的问题总结
- css3属性整理
- POJ 2373 Yogurt factory
- QT server服务端如何判断客户端断开连接
- LeetCode 448. Find All Numbers Disappeared in an Array (在数组中找到没有出现的数字)
- Linux 下的分屏利器-tmux安装、原理及使用
- 关于破解visualsvn 我这里是版本是5.2.1
- Oracle备份恢复简单过程以及中间的坑.
- (转)Python3之pickle模块
- Java 使用getClass().getResourceAsStream()方法获取资源
- 初识WebAPI
热门文章
- 【devexpress】spinEdit控件如何设置只能输入两位小数
- 打印九九乘法表,打印金字塔-java
- 【翻译】rocksdb调试指引
- 【数据库】Postgresql/PG-编写函数实现字段对应加备注
- 【数据库】SQL-随机生成区间内数值、日期、字符串,mock数据
- 【zookeeper】Zookeeper相关概念、重难点(myid)、语法、使用、工具
- 【基础语法规范】【函数式编程、字符串分割】BC6:输出输入的第二个整数
- 【每日一题】【暴力&;双指针&;动态规划】42. 接雨水-211130/220214
- 【每日一题】【二分mid&;贪心】2022年2月8日-NC163 最长上升子序列(一)
- 5V升压12.6V芯片电路图,三节锂电池充电