Nacos注册中心源码分析二(服务端注册底层源码分析)

  上篇文章介绍了,nacos客户端底层是如何实现服务注册的,通过源码分析我们知道在客户端启动的时候,通过NacosAutoServiceRegistration这个类继承了ApplicaqtionListener,最终在spring容器启动的时候会回调

onApplicationEvent这个方法,最终会通过发起http请求调用的Nacos服务端提供的接口,那么这篇文章我们来聊一下Nacos服务端是如何实现服务注册的逻辑的。

一,源码下载

  首先从github上面将Naocs的代码下载下来,然后通过idea导入进去,编译完以后,就可以开始看源码了,

可以看到Nacos分了很多模块,我们今天要说的服务端服务注册的逻辑就是在这个naming模块里面,

 

源码的入口就是这个InstanceController里面的register方法,这里面就Nacos服务注册的源码实现,下面我们一起来看看服务端注册逻辑是如何实现的,首先就是一些参数的校验的分支逻辑,我们主要看主干逻辑,我们跟进去serviceManager.registerInstance(namespaceId, serviceName, instance)方法

 

/**
 * Register an instance to a service in AP mode.
 *
 * <p>This method creates service or cluster silently if they don't exist.
 *
 * @param namespaceId id of namespace
 * @param serviceName service name
 * @param instance    instance to register
 * @throws Exception any error occurred in the process
 */
public void registerInstance(String namespaceId, String serviceName, Instance instance) throws NacosException {
    
    createEmptyService(namespaceId, serviceName, instance.isEphemeral());
    
    Service service = getService(namespaceId, serviceName);
    
    if (service == null) {
        throw new NacosException(NacosException.INVALID_PARAM,
                "service not found, namespace: " + namespaceId + ", service: " + serviceName);
    }
    
    addInstance(namespaceId, serviceName, instance.isEphemeral(), instance);
}

首先createEmptyService会创建一个空的Service对象,也就是将服务提供者的一些信息封装到这个Service对象里面,下面这个addInstance方法就是重点的方法,看这个方法的名字也很好理解,添加实例信息,继续跟进去这个方法里面

/**
 * Add instance to service.
 *
 * @param namespaceId namespace
 * @param serviceName service name
 * @param ephemeral   whether instance is ephemeral
 * @param ips         instances
 * @throws NacosException nacos exception
 */
public void addInstance(String namespaceId, String serviceName, boolean ephemeral, Instance... ips)
        throws NacosException {
    
    String key = KeyBuilder.buildInstanceListKey(namespaceId, serviceName, ephemeral);
    
    Service service = getService(namespaceId, serviceName);
    
    synchronized (service) {
        List<Instance> instanceList = addIpAddresses(service, ephemeral, ips);
        
        Instances instances = new Instances();
        instances.setInstanceList(instanceList);
        
        consistencyService.put(key, instances);
    }
}

首先会生成一个存放servie的key。然后拿到当前所有的服务实例,掉用接口consistencyService.put(key, instances)方法,这个接口有以下的实现类

这里如何知道这个put方法会调用到哪个实现类的方法呢,这里我们点击这个consistencyService,

可以看做这个接口注入的时候,指定了名称,而这几个实现类,里面只有DelegateConsistencyServiceImpl这个实现类,指定的Bean的名称也是这个consitencyDelegate

也就是最终会调用DelegateCoonsistencyServiceImpl里面的put方法

@Override
public void put(String key, Record value) throws NacosException {
    mapConsistencyService(key).put(key, value);
}

这个mapConsistencyService(key)会根据是否临时实例,来判断选择对应的实现类

private ConsistencyService mapConsistencyService(String key) {
    return KeyBuilder.matchEphemeralKey(key) ? ephemeralConsistencyService : persistentConsistencyService;
}

而Nacos默认是使用临时实例实现了,也就是AP模式的架构

可以看到这个DistroConsistencyServiceImpl实现了EphemeralConsistencyService接口,最后会调用D

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Nacos 注册服务源码涉及的主要类有以下几个: 1. NamingService:命名服务接口,定义了注册、注销、查询服务的方法。 2. NamingServiceImpl:命名服务的实现类,实现了 NamingService 接口。 3. ServerListManager:服务列表管理器,负责管理服务列表。 4. Instance:服务实例对象,包含了服务实例的基本信息。 5. Service:服务对象,包含了服务的基本信息和服务实例列表。 6. ServiceInfo:服务信息对象,包含了所有服务和服务实例的信息。 7. ServerProxy:服务代理类,负责与服务注册中心通信。 下面对这些类进行详细的分析。 1. NamingService 命名服务接口,定义了注册、注销、查询服务的方法。 ```java public interface NamingService { /** * 注册一个服务实例 */ void registerInstance(String serviceName, Instance instance) throws NacosException; /** * 注销一个服务实例 */ void deregisterInstance(String serviceName, Instance instance) throws NacosException; /** * 查询一个服务的所有实例 */ List<Instance> getAllInstances(String serviceName) throws NacosException; /** * 监听一个服务的实例变化 */ void subscribe(String serviceName, EventListener listener) throws NacosException; /** * 取消监听一个服务的实例变化 */ void unsubscribe(String serviceName, EventListener listener) throws NacosException; } ``` 2. NamingServiceImpl 命名服务的实现类,实现了 NamingService 接口。 ```java public class NamingServiceImpl implements NamingService { private ServerProxy serverProxy; public NamingServiceImpl(ServerProxy serverProxy) { this.serverProxy = serverProxy; } @Override public void registerInstance(String serviceName, Instance instance) throws NacosException { serverProxy.registerService(serviceName, instance); } @Override public void deregisterInstance(String serviceName, Instance instance) throws NacosException { serverProxy.deregisterService(serviceName, instance); } @Override public List<Instance> getAllInstances(String serviceName) throws NacosException { return serverProxy.getAllInstances(serviceName); } @Override public void subscribe(String serviceName, EventListener listener) throws NacosException { serverProxy.subscribe(serviceName, listener); } @Override public void unsubscribe(String serviceName, EventListener listener) throws NacosException { serverProxy.unsubscribe(serviceName, listener); } } ``` 3. ServerListManager 服务列表管理器,负责管理服务列表。 ```java public class ServerListManager { private DiscoveryConfig config; private List<ServerInfo> serverList = new ArrayList<>(); private AtomicInteger index = new AtomicInteger(0); public ServerListManager(DiscoveryConfig config) { this.config = config; initServerList(); } private void initServerList() { // 读取配置文件中的服务地址列表 String serverListStr = config.getServerList(); String[] serverArr = serverListStr.split(","); for (int i = 0; i < serverArr.length; i++) { String[] serverInfoArr = serverArr[i].split(":"); String ip = serverInfoArr[0]; int port = Integer.parseInt(serverInfoArr[1]); ServerInfo serverInfo = new ServerInfo(ip, port); serverList.add(serverInfo); } } public ServerInfo getNextServer() { // 轮询获取服务地址 int i = index.getAndIncrement() % serverList.size(); return serverList.get(i); } } ``` 4. Instance 服务实例对象,包含了服务实例的基本信息。 ```java public class Instance { private String serviceName; private String ip; private int port; public Instance(String serviceName, String ip, int port) { this.serviceName = serviceName; this.ip = ip; this.port = port; } public String getServiceName() { return serviceName; } public void setServiceName(String serviceName) { this.serviceName = serviceName; } public String getIp() { return ip; } public void setIp(String ip) { this.ip = ip; } public int getPort() { return port; } public void setPort(int port) { this.port = port; } } ``` 5. Service 服务对象,包含了服务的基本信息和服务实例列表。 ```java public class Service { private String name; private List<Instance> instances = new ArrayList<>(); public Service(String name) { this.name = name; } public String getName() { return name; } public List<Instance> getInstances() { return instances; } public void addInstance(Instance instance) { instances.add(instance); } public void removeInstance(Instance instance) { instances.remove(instance); } } ``` 6. ServiceInfo 服务信息对象,包含了所有服务和服务实例的信息。 ```java public class ServiceInfo { private Map<String, Service> serviceMap = new HashMap<>(); public void addService(Service service) { serviceMap.put(service.getName(), service); } public void removeService(Service service) { serviceMap.remove(service.getName()); } public List<Service> getServices() { return new ArrayList<>(serviceMap.values()); } } ``` 7. ServerProxy 服务代理类,负责与服务注册中心通信。 ```java public class ServerProxy { private ServerListManager serverListManager; public ServerProxy(ServerListManager serverListManager) { this.serverListManager = serverListManager; } public void registerService(String serviceName, Instance instance) throws NacosException { ServerInfo serverInfo = serverListManager.getNextServer(); try (Socket socket = new Socket(serverInfo.getIp(), serverInfo.getPort())) { // 向注册中心发送注册请求 // ... } catch (IOException e) { // 处理异常 } } public void deregisterService(String serviceName, Instance instance) throws NacosException { // 向注册中心发送注销请求 // ... } public List<Instance> getAllInstances(String serviceName) throws NacosException { // 从注册中心获取服务实例列表 // ... return instances; } public void subscribe(String serviceName, EventListener listener) throws NacosException { // 向注册中心发送订阅请求 // ... } public void unsubscribe(String serviceName, EventListener listener) throws NacosException { // 向注册中心发送取消订阅请求 // ... } } ``` 以上就是 Nacos 注册服务源码分析的内容。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值