nacos官方地址:https://nacos.io/zh-cn/

大家可以看一下nacos的中文手册以及官方源码,博主就不带领大家快速入门 了,官方文档中都有而且非常标准,比其他博客写的好多了并且还是实时更新的。

先看一下博主给大家画的流程图,掌握一下大概的基本流程,好理解,博主给大家讲源码:

https://www.processon.com/view/link/5f7e895be0b34d0711f65178

nacos最主要的功能就是服务注册及发现,那它到底是如何实现的呢?博主用的springboot开发的,所以直接就去找nacos的jar包下的spring.factories,这是每个要自动注入的服务jar的必备文件,我们来看一下

  里面有很多的自动配置类,我们只看一下NacosServiceRegistryAutoConfiguration,该类主要做的是服务注册的相关事宜,如果大家第一次看源码的话不知道该如何下手,最笨的方法就是一个一个看。总会找到的,最好要找名字看起来就像自己找的,因为这是阿里巴巴开发的,他们是有java开发规范的,要做到见名知意。

 1  /**
2 * @author xiaojing
3 * @author <a href="mailto:mercyblitz@gmail.com">Mercy</a>
4 */
5 @Configuration(proxyBeanMethods = false)
6 @EnableConfigurationProperties
7 @ConditionalOnNacosDiscoveryEnabled
8 @ConditionalOnProperty(value = "spring.cloud.service-registry.auto-registration.enabled",
9 matchIfMissing = true)
10 @AutoConfigureAfter({ AutoServiceRegistrationConfiguration.class,
11 AutoServiceRegistrationAutoConfiguration.class,
12 NacosDiscoveryAutoConfiguration.class })
13 public class NacosServiceRegistryAutoConfiguration {
14
15 @Bean
16 public NacosServiceRegistry nacosServiceRegistry(
17 NacosDiscoveryProperties nacosDiscoveryProperties) {
18 return new NacosServiceRegistry(nacosDiscoveryProperties);
19 }
20
21 @Bean
22 @ConditionalOnBean(AutoServiceRegistrationProperties.class)
23 public NacosRegistration nacosRegistration(
24 NacosDiscoveryProperties nacosDiscoveryProperties,
25 ApplicationContext context) {
26 return new NacosRegistration(nacosDiscoveryProperties, context);
27 }
28
29 @Bean
30 @ConditionalOnBean(AutoServiceRegistrationProperties.class)
31 public NacosAutoServiceRegistration nacosAutoServiceRegistration(
32 NacosServiceRegistry registry,
33 AutoServiceRegistrationProperties autoServiceRegistrationProperties,
34 NacosRegistration registration) {
35 //这里才是自动注入的类,我们需要进去看一下
36 return new NacosAutoServiceRegistration(registry,
37 autoServiceRegistrationProperties, registration);
38 }
39
40 }
 1 public class NacosAutoServiceRegistration
2 extends AbstractAutoServiceRegistration<Registration> {
3
4 private static final Logger log = LoggerFactory
5 .getLogger(NacosAutoServiceRegistration.class);
6
7 private NacosRegistration registration;
8
9 public NacosAutoServiceRegistration(ServiceRegistry<Registration> serviceRegistry,
10 AutoServiceRegistrationProperties autoServiceRegistrationProperties,
11 NacosRegistration registration) {
12 super(serviceRegistry, autoServiceRegistrationProperties);
13 this.registration = registration;
14 }
15
16 @Deprecated
17 public void setPort(int port) {
18 getPort().set(port);
19 }
20
21 @Override
22 protected NacosRegistration getRegistration() {
23 if (this.registration.getPort() < 0 && this.getPort().get() > 0) {
24 this.registration.setPort(this.getPort().get());
25 }
26 Assert.isTrue(this.registration.getPort() > 0, "service.port has not been set");
27 return this.registration;
28 }
29
30 @Override
31 protected NacosRegistration getManagementRegistration() {
32 return null;
33 }
34
35 @Override
36 protected void register() {
37 if (!this.registration.getNacosDiscoveryProperties().isRegisterEnabled()) {
38 log.debug("Registration disabled.");
39 return;
40 }
41 if (this.registration.getPort() < 0) {
42 this.registration.setPort(getPort().get());
43 }
44 super.register();
45 }
46
47 @Override
48 protected void registerManagement() {
49 if (!this.registration.getNacosDiscoveryProperties().isRegisterEnabled()) {
50 return;
51 }
52 super.registerManagement();
53
54 }
55
56 @Override
57 protected Object getConfiguration() {
58 return this.registration.getNacosDiscoveryProperties();
59 }
60
61 @Override
62 protected boolean isEnabled() {
63 return this.registration.getNacosDiscoveryProperties().isRegisterEnabled();
64 }
65
66 @Override
67 @SuppressWarnings("deprecation")
68 protected String getAppName() {
69 String appName = registration.getNacosDiscoveryProperties().getService();
70 return StringUtils.isEmpty(appName) ? super.getAppName() : appName;
71 }
72
73 }
看到这里,大家可能有点懵,我应该找那个方法呢?大家可以看一下构造器中,引用了父类,当我们看到父类的时候,发现父类实现了ApplicationListener,这个类大家不陌生,系统会自动执行其onApplicationEvent而正好我们的类实现了这个方法,最后我们发现了应该看register() 方法,大家可以来看看这个方法的实现
 1 @Override
2 public void register(Registration registration) {
3
4 if (StringUtils.isEmpty(registration.getServiceId())) {
5 log.warn("No service to register for nacos client...");
6 return;
7 }
8 //看到这两个参数,大家看过nacos文档就知道这两个参数要干啥用了
9 String serviceId = registration.getServiceId();
10 String group = nacosDiscoveryProperties.getGroup();
11 //要发送的实例对象,具体的属性,大家可以看看这个方法
12 Instance instance = getNacosInstanceFromRegistration(registration);
13
14 try {
15 //走这里,是真正的注册服务
16 namingService.registerInstance(serviceId, group, instance);
17 log.info("nacos registry, {} {} {}:{} register finished", group, serviceId,
18 instance.getIp(), instance.getPort());
19 }
20 catch (Exception e) {
21 log.error("nacos registry, {} register failed...{},", serviceId,
22 registration.toString(), e);
23 // rethrow a RuntimeException if the registration is failed.
24 // issue : https://github.com/alibaba/spring-cloud-alibaba/issues/1132
25 rethrowRuntimeException(e);
26 }
27 }
 1     @Override
2 public void registerInstance(String serviceName, String groupName, Instance instance) throws NacosException {
3 //是否是临时的,这个参数默认是临时实例,这个是临时或者是否持久很重要,先记住
4 if (instance.isEphemeral()) {
5 //填写各种心跳信息,毕竟服务需要往nacos发送心跳,nacos才能知道该服务是否还存活
6 BeatInfo beatInfo = new BeatInfo();
7 beatInfo.setServiceName(NamingUtils.getGroupedName(serviceName, groupName));
8 beatInfo.setIp(instance.getIp());
9 beatInfo.setPort(instance.getPort());
10 beatInfo.setCluster(instance.getClusterName());
11 beatInfo.setWeight(instance.getWeight());
12 beatInfo.setMetadata(instance.getMetadata());
13 beatInfo.setScheduled(false);
14 beatInfo.setPeriod(instance.getInstanceHeartBeatInterval());
15 //看一下客服端是如何发送心跳
16 beatReactor.addBeatInfo(NamingUtils.getGroupedName(serviceName, groupName), beatInfo);
17 }
18 //注册服务
19 serverProxy.registerService(NamingUtils.getGroupedName(serviceName, groupName), groupName, instance);
20 }
 1         @Override
2 public void run() {
3 if (beatInfo.isStopped()) {
4 return;
5 }
6 long nextTime = beatInfo.getPeriod();
7 try {
8 //会调用nacos服务端的/instance/beat接口
9 JSONObject result = serverProxy.sendBeat(beatInfo, BeatReactor.this.lightBeatEnabled);
10 long interval = result.getIntValue("clientBeatInterval");
11 boolean lightBeatEnabled = false;
12 if (result.containsKey(CommonParams.LIGHT_BEAT_ENABLED)) {
13 lightBeatEnabled = result.getBooleanValue(CommonParams.LIGHT_BEAT_ENABLED);
14 }
15 BeatReactor.this.lightBeatEnabled = lightBeatEnabled;
16 if (interval > 0) {
17 nextTime = interval;
18 }
19 int code = NamingResponseCode.OK;
20 if (result.containsKey(CommonParams.CODE)) {
21 code = result.getIntValue(CommonParams.CODE);
22 }
23 //如果找不到服务,则进行注册服务
24 if (code == NamingResponseCode.RESOURCE_NOT_FOUND) {
25 Instance instance = new Instance();
26 instance.setPort(beatInfo.getPort());
27 instance.setIp(beatInfo.getIp());
28 instance.setWeight(beatInfo.getWeight());
29 instance.setMetadata(beatInfo.getMetadata());
30 instance.setClusterName(beatInfo.getCluster());
31 instance.setServiceName(beatInfo.getServiceName());
32 instance.setInstanceId(instance.getInstanceId());
33 instance.setEphemeral(true);
34 try {
35 //该方法封装好参数后,会调用/instance接口
36 serverProxy.registerService(beatInfo.getServiceName(),
37 NamingUtils.getGroupName(beatInfo.getServiceName()), instance);
38 } catch (Exception ignore) {
39 }
40 }
41 } catch (NacosException ne) {
42 NAMING_LOGGER.error("[CLIENT-BEAT] failed to send beat: {}, code: {}, msg: {}",
43 JSON.toJSONString(beatInfo), ne.getErrCode(), ne.getErrMsg());
44
45 }
46 //定时线程池不会一直循环进行调用,所以每次执行完之后会继续添加定时任务进行发送心跳
47 executorService.schedule(new BeatTask(beatInfo), nextTime, TimeUnit.MILLISECONDS);
48 }

  看完发送心跳的方法,我们该看注册服务的方法了:serverProxy.registerService(NamingUtils.getGroupedName(serviceName, groupName), groupName, instance);其实就是发送心跳后,找不到服务时调用的/instance接口。

  到此,我们的服务就会注册到nacos服务端中,客户端的代码就是如此,还是挺简单的,我们下一篇就会带大家走进服务端的代码。


Nacos(一)源码分析Nacos注册示例流程的更多相关文章

  1. Yii2 源码分析 入口文件执行流程

    Yii2 源码分析  入口文件执行流程 1. 入口文件:web/index.php,第12行.(new yii\web\Application($config)->run()) 入口文件主要做4 ...

  2. Solr4.8.0源码分析(5)之查询流程分析总述

    Solr4.8.0源码分析(5)之查询流程分析总述 前面已经写到,solr查询是通过http发送命令,solr servlet接受并进行处理.所以solr的查询流程从SolrDispatchsFilt ...

  3. (转)linux内存源码分析 - 内存回收(整体流程)

    http://www.cnblogs.com/tolimit/p/5435068.html------------linux内存源码分析 - 内存回收(整体流程) 概述 当linux系统内存压力就大时 ...

  4. HDFS源码分析DataXceiver之整体流程

    在<HDFS源码分析之DataXceiverServer>一文中,我们了解到在DataNode中,有一个后台工作的线程DataXceiverServer.它被用于接收来自客户端或其他数据节 ...

  5. JVM源码分析之JVM启动流程

      原创申明:本文由公众号[猿灯塔]原创,转载请说明出处标注 “365篇原创计划”第十四篇. 今天呢!灯塔君跟大家讲: JVM源码分析之JVM启动流程 前言: 执行Java类的main方法,程序就能运 ...

  6. Nacos(二)源码分析Nacos服务端注册示例流程

    上回我们讲解了客户端配置好nacos后,是如何进行注册到服务器的,那我们今天来讲解一下服务器端接收到注册实例请求后会做怎么样的处理. 首先还是把博主画的源码分析图例发一下,让大家对整个流程有一个大概的 ...

  7. Java并发包中Semaphore的工作原理、源码分析及使用示例

    1. 信号量Semaphore的介绍 我们以一个停车场运作为例来说明信号量的作用.假设停车场只有三个车位,一开始三个车位都是空的.这时如果同时来了三辆车,看门人允许其中它们进入进入,然后放下车拦.以后 ...

  8. Hadoop RCFile存储格式详解(源码分析、代码示例)

    RCFile   RCFile全称Record Columnar File,列式记录文件,是一种类似于SequenceFile的键值对(Key/Value Pairs)数据文件.   关键词:Reco ...

  9. springMVC源码分析--AbstractHandlerMethodMapping注册url和HandlerMethod对应关系(十一)

    在上一篇博客springMVC源码分析--AbstractHandlerMethodMapping获取url和HandlerMethod对应关系(十)中我们简单地介绍了获取url和HandlerMet ...

  10. illuminate/routing 源码分析之注册路由

    我们知道,在 Laravel 世界里,外界传进来一个 Request 时,会被 Kernel 处理并返回给外界一个 Response.Kernel 在处理 Request 时,会调用 illumina ...

随机推荐

  1. jvm系列(二)jvm垃圾收集器与内存分配策略

    众所周知,在java语言中,内存分配和回收是由jvm自动管理的.因此内存的分配和回收也是jvm三大功能之一.垃圾收集器(GC)需要完成三件事情: 哪些内存需要回收? 什么时候进行回收? 如何回收? 本 ...

  2. 思维导图学《JVM 虚拟机规范》

    目录 工具 虚拟机实现 class 文件结构 字节码指令 其他 虚拟机结构 公众号 coding 笔记.点滴记录,以后的文章也会同步到公众号(Coding Insight)中,希望大家关注_ 公众号 ...

  3. java12(eclipse断点调试)

    选择结构switch 1.格式: switch(整型数据){ case 值A:System.out.println("");break; case 值B:System.out.pr ...

  4. Django 的缓存机制

    一 缓存介绍: 在动态网站中,用户所有的请求,服务器都会去数据库中进行相应的增,删,查,改,渲染模板,执行业务逻辑,最后生成用户看到的页面. 当一个网站的用户访问量很大的时候,每一次的的后台操作,都会 ...

  5. day1(Django)

    1,web项目工作流程 1.1 了解web程序工作流程 1.2 django生命周期   2,django介绍 目的:了解Django框架的作用和特点作用: 简便.快速的开发数据库驱动的网站 Djan ...

  6. 这可能是最为详细的Docker入门总结

    写在前面 毕设是关于区块链的,自然就用到了docker,感觉到了docker的强大.学习源于总结,所以找了一些资料,这篇文章原作写的不错,看了好多遍哈哈. 这可能是最为详细的Docker入门总结 市面 ...

  7. 第8.2节 Python类的__init__方法深入剖析:构造方法案例详解

    前面一节介绍了构造方法定义的语法,并进行了语法解释说明,本节将通过案例来说明构造方法参数传递及返回值的情况. 一.    案例说明 本节定义一个汽车类,它有四个实例变量:wheelcount, pow ...

  8. 第7.26节 Python中的@property装饰器定义属性访问方法getter、setter、deleter 详解

    第7.26节 Python中的@property装饰器定义属性访问方法getter.setter.deleter 详解 一.    引言 Python中的装饰器在前面接触过,老猿还没有深入展开介绍装饰 ...

  9. PyQt(Python+Qt)学习随笔:QListWidget获取当前选中项的selectedItems方法

    老猿Python博文目录 专栏:使用PyQt开发图形界面Python应用 老猿Python博客地址 QListWidget的selectedItems方法返回列表部件中所有选中项的一个列表,调用语法如 ...

  10. 第十四章 web前端开发小白学爬虫

    老猿从事IT开发快三十年了,接触互联网也很久了,但自己没有做过web前端开发,只知道与前端开发相关的一些基本概念,如B/S架构.html标签.js脚本.css样式.xml解析.cookies.http ...