草庐IT

【微服务~原始真解】Spring Cloud —— Eureka Client原码解析(二)

桃花键神 2023-07-09 原文

🔎这里是【秒懂·云原生】,关注我学习云原生不迷路
👍如果对你有帮助,给博主一个免费的点赞以示鼓励
欢迎各位🔎点赞👍评论收藏⭐️

👀专栏介绍

【秒懂·云原生】 目前主要更新微服务,一起学习一起进步。

👀本期介绍

主要介绍Spring Cloud —— Eureka Client原码解析(二)

文章目录

服务发现客户端

为了对Eureka Client的执行原理进行讲解,首先需要对服务发现客户端com.netflix.discover.DiscoveryClient职能以及相关类进行讲解,它负责了与EurekaServer交互的关键逻辑。

1.DiscoveryClient职责

DiscoveryClient是Eureka Client的核心类,包括与Eureka Server交互的关键逻辑,具备了以下职能:

  • 注册服务实例到Eureka Server中;
  • 发送心跳更新与Eureka Server的租约;
  • 在服务关闭时从Eureka Server中取消租约,服务下线;
  • 查询在Eureka Server中注册的服务实例列表。

2.DiscoveryClient类结构

DiscoverClient的核心类图如图所示。

DiscoveryClient继承了LookupService接口,LookupService作用是发现活跃的服务实例,主要方法如下:

//LookupService. java
public interface LookupService<T> {
//根据服务实例注册的appName来获取封装有相同appName的服务实例信息容器Application getApplication(String appName) ;
//返回当前注册表中所有的服务实例信息
Applications getApplications () ;//根据服务实例的id获取服务实例信息
List<InstanceInfo> getInstancesById(String id) ;
}

Application持有服务实例信息列表,它可以理解成同一个服务的集群信息,这些服务实例都挂在同一个服务名appName下。InstanceInfo代表一个服务实例信息。
Application部分代码如下:

//Application. java
public class Application {
private static Random shuffleRandom = new Random() ;
//服务名
private String name;
@XStream0mitField
private volatile boolean isDirty = false;
@XStreamImplicit
private final Set<InstanceInfo> instances;
private final AtomicReference <List<InstanceInfo>> shuffledInstances;
private final Map<StringInstanceInfo> instancesMap;
}

为了保证原子性操作,Application中对Instancelnfo的操作都是同步操作。

Applications是注册表中所有服务实例信息的集合,里面的操作大多也是同步操作。

EurekaClient继承了LookupService接口,为DiscoveryClient提供了一个上层接口,目的是方便从Eureka 1.x到Eureka 2.x(已停止开发)的升级过渡。EurekaClient接口属于比较稳定的接口,即使在下一阶段也会被保留。

EurekaCient在LookupService的基础上扩充了更多的接口,提供了更丰富的获取服务实例的方式,主要有:

  • 提供了多种方式获取Instancelnfo,例如根据区域、Eureka Server地址等获取。
  • 提供了本地客户端(所处的区域、可用区等)的数据,这部分与AWS密切相关。
  • 提供了为客户端注册和获取健康检查处理器的能力。

除去查询相关的接口,我们主要关注EurekaClient中以下两个接口,代码如下所示:

// EurekaClient.java
//为Eureka Client注册健康检查处理器
public void registerHealthCheck(HealthCheckHandler healthCheckHandler);
//为Eureka Client注册一个EurekaEventListener(事件监听器)
//监听Client服务实例信息的更新
public void registerEventListener(EurekaEventListener eventListener);

Eureka Server一般通过心跳(heartbeats)来识别一个实例的状态。EurekaClient中存在一个定时任务定时通过HealthCheckHandler检测当前Client的状态,如果Client的状态发生改变,将会触发新的注册事件,更新Eureka Server的注册表中该服务实例的相关信息。HealthCheckHandler的代码如下所示:

// HealthCheckHandler.java
public interface HealthCheckHandler {
InstanceInfo.InstanceStatus getStatus (InstanceInfo.InstanceStatus currentStatus) ;
}

HealthCheckHandler接口的代码如上所示,其在spring-cloud-netflix-eureka-client中的实现类为EurekaHealthCheckHandler,主要组合了spring-boot-actuator中的HealthAggregator和HealthIlndicator,以实现对Spring Boot应用的状态检测。
Eureka中的事件模式属于观察者模式,事件监听器将监听Client的服务实例信息变化,触发对应的处理事件,下图为Eureka事件的类图:

3.DiscoveryClient

DiscoveryClient可以说是Eureka Client的核心类,负责了与Eureka Server交互的关键逻辑,具备了以下的职能:

  • 注册服务实例到Eureka Server中;
  • 更新与Eureka Server的契约;
  • 在服务关闭时从Eureka Server中取消契约;
  • 查询在Eureka Server中注册的服务/实例的列表。

DiscoverClient的核心类图如下:

DiscoveryClient的顶层接口为LookupService,主要的目的是为了发现活跃中的服务实例。

public interface LookupService<T> {
 
	//根据服务实例注册的appName来获取,获取一个封装有相同appName的服务实例信息的容器
   Application getApplication(String appName);
	//返回当前注册的所有的服务实例信息
   Applications getApplications();
   	//根据服务实例的id获取
   	List<InstanceInfo> getInstancesById(String id);
   //获取下一个可能的Eureka Server来处理当前对注册表信息的处理,一般是通过循环的方式来获取下一个Server
   InstanceInfo getNextServerFromEureka(String virtualHostname, boolean secure);
}

Application中持有一个特定应用的多个实例的列表,可以理解成同一个服务的集群信息,它们都挂在同一个服务名appName下,InstanceInfo代表一个服务实例,部分代码如下:

public class Application {
    
    private static Random shuffleRandom = new Random();
 
    //服务名
    private String name;
 
    @XStreamOmitField
    private volatile boolean isDirty = false;
 
    @XStreamImplicit
    private final Set<InstanceInfo> instances;
 
    private final AtomicReference<List<InstanceInfo>> shuffledInstances;
 
    private final Map<String, InstanceInfo> instancesMap;
    
    .....
}

为了保证原子性操作以及数据的唯一性,防止脏数据,Application中对InstanceInfo的操作都是同步操作,感受一下Application.addInstance方法。

public void addInstance(InstanceInfo i) {
	instancesMap.put(i.getId(), i);
	synchronized (instances) {
	instances.remove(i);
	instances.add(i);
	isDirty = true;
	}
}

通过同步代码块,保证每次只有有一个线程对instances进行修改,同时注意instancesMap采用的是ConcurrentHashMap实现,保证了原子性的操作,所以不需要通过同步代码块进行控制。

Applications中代表的是Eureka Server中已注册的服务实例的集合信息,主要是对Application的封装,里面的操作大多也是的同步操作。

EurekaClient继承了LookupService接口,为DiscoveryClient提供了一个上层的接口,目的是试图方便从eureka 1.x 到eureka 2.x 的过渡,这说明EurekaClient这个接口属于比较稳定的接口,即使在下一大阶段也会被依旧保留。

EurekaCient在LookupService的基础上扩充了更多的接口,提供了更丰富的获取服务实例的功能,主要有:

  • 提供了多种的方式获取InstanceInfo,例如根据region,Eureka Server地址等获取;
  • 提供了本地客户端(位于的区域,可用区等)的数据,这部分与AWS密切相关;
  • 提供了为客户端注册和获取健康检查处理器;

除去查询相关的接口,关注EurekaClient中的以下两个接口:

    // 为Eureka Client注册健康检查处理器
    // 一旦注册,客户端将通过调用新注册的健康检查处理器来对注册中instanceInfo
    // 进行一个按需更新,随后按照eurekaclientconfig.getinstanceinforeplicationintervalseconds()
    // 中配置的指定时间调用HealthCheckHandler
    public void registerHealthCheck(HealthCheckHandler healthCheckHandler);
 
    // 为eureka client注册一个EurekaEventListener(事件监听器)
    // 一旦注册,当eureka client的内部状态发生改变的时候,将会调用EurekaEventListener.onEvent()
    // 触发一定的事件。可以通过这种方式监听client的更新而非通过轮询的方式询问client
    public void registerEventListener(EurekaEventListener eventListener);

Eureka Server一般通过心跳(heartbeats)来识别一个实例的状态。Eureka Client中存在一个定时任务定时通过HealthCheckHandler检测当前client的状态,如果client的状态发生改变,将会触发新的注册事件,同步Eureka Server的注册表中该服务实例的相关信息。

public interface HealthCheckHandler {
    InstanceInfo.InstanceStatus getStatus(InstanceInfo.InstanceStatus currentStatus);
}

在spring-cloud-netflix-eureka-client中实现了这个的接口,EurekaHealthCheckHandler,主要的组合了spring-boot-actuator中的HealthAggregator和HealthIndicator实现了对spring-boot应用的状态检测。
主要有以下的状态:

public enum InstanceStatus {
	UP, // 可以接受服务请求
	DOWN, // 无法发送流量-健康检查失败
	STARTING, // 正在启动,无法发送流量
	OUT_OF_SERVICE, // 服务关闭,不接受流量
	UNKNOWN; // 未知状态
    }

Eureka中的事件模式,这是一个很明显的观察者模式,以下为它的类图类图:

有关【微服务~原始真解】Spring Cloud —— Eureka Client原码解析(二)的更多相关文章

  1. Ruby 解析字符串 - 2

    我有一个字符串input="maybe(thisis|thatwas)some((nice|ugly)(day|night)|(strange(weather|time)))"Ruby中解析该字符串的最佳方法是什么?我的意思是脚本应该能够像这样构建句子:maybethisissomeuglynightmaybethatwassomenicenightmaybethiswassomestrangetime等等,你明白了......我应该一个字符一个字符地读取字符串并构建一个带有堆栈的状态机来存储括号值以供以后计算,还是有更好的方法?也许为此目的准备了一个开箱即用的库?

  2. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  3. ruby - 用逗号、双引号和编码解析 csv - 2

    我正在使用ruby​​1.9解析以下带有MacRoman字符的csv文件#encoding:ISO-8859-1#csv_parse.csvName,main-dialogue"Marceu","Giveittohimóhe,hiswife."我做了以下解析。require'csv'input_string=File.read("../csv_parse.rb").force_encoding("ISO-8859-1").encode("UTF-8")#=>"Name,main-dialogue\r\n\"Marceu\",\"Giveittohim\x97he,hiswife.\"\

  4. ruby-on-rails - 我更新了 ruby​​ gems,现在到处都收到解析树错误和弃用警告! - 2

    简而言之错误:NOTE:Gem::SourceIndex#add_specisdeprecated,useSpecification.add_spec.Itwillberemovedonorafter2011-11-01.Gem::SourceIndex#add_speccalledfrom/opt/local/lib/ruby/site_ruby/1.8/rubygems/source_index.rb:91./opt/local/lib/ruby/gems/1.8/gems/rails-2.3.8/lib/rails/gem_dependency.rb:275:in`==':und

  5. Observability:从零开始创建 Java 微服务并监控它 (二) - 2

    这篇文章是继上一篇文章“Observability:从零开始创建Java微服务并监控它(一)”的续篇。在上一篇文章中,我们讲述了如何创建一个Javaweb应用,并使用Filebeat来收集应用所生成的日志。在今天的文章中,我来详述如何收集应用的指标,使用APM来监控应用并监督web服务的在线情况。源码可以在地址 https://github.com/liu-xiao-guo/java_observability 进行下载。摄入指标指标被视为可以随时更改的时间点值。当前请求的数量可以改变任何毫秒。你可能有1000个请求的峰值,然后一切都回到一个请求。这也意味着这些指标可能不准确,你还想提取最小/

  6. ruby - 用 YAML.load 解析 json 安全吗? - 2

    我正在使用ruby2.1.0我有一个json文件。例如:test.json{"item":[{"apple":1},{"banana":2}]}用YAML.load加载这个文件安全吗?YAML.load(File.read('test.json'))我正在尝试加载一个json或yaml格式的文件。 最佳答案 YAML可以加载JSONYAML.load('{"something":"test","other":4}')=>{"something"=>"test","other"=>4}JSON将无法加载YAML。JSON.load("

  7. ruby - 如何使用 Nokogiri 解析纯 HTML 表格? - 2

    我想用Nokogiri解析HTML页面。页面的一部分有一个表,它没有使用任何特定的ID。是否可以提取如下内容:Today,3,455,34Today,1,1300,3664Today,10,100000,3444,Yesterday,3454,5656,3Yesterday,3545,1000,10Yesterday,3411,36223,15来自这个HTML:TodayYesterdayQntySizeLengthLengthSizeQnty345534345456563113003664354510001010100000344434113622315

  8. python - 帮我找到合适的 ruby​​/python 解析器生成器 - 2

    我使用的第一个解析器生成器是Parse::RecDescent,它的指南/教程很棒,但它最有用的功能是它的调试工具,特别是tracing功能(通过将$RD_TRACE设置为1来激活)。我正在寻找可以帮助您调试其规则的解析器生成器。问题是,它必须用python或ruby​​编写,并且具有详细模式/跟踪模式或非常有用的调试技术。有人知道这样的解析器生成器吗?编辑:当我说调试时,我并不是指调试python或ruby​​。我指的是调试解析器生成器,查看它在每一步都在做什么,查看它正在读取的每个字符,它试图匹配的规则。希望你明白这一点。赏金编辑:要赢得赏金,请展示一个解析器生成器框架,并说明它的

  9. ruby - 如何用 Nokogiri 解析连续的标签? - 2

    我有这样的HTML代码:Label1Value1Label2Value2...我的代码不起作用。doc.css("first").eachdo|item|label=item.css("dt")value=item.css("dd")end显示所有首先标记,然后标记标签,我需要“标签:值” 最佳答案 首先,您的HTML应该有和中的元素:Label1Value1Label2Value2...但这不会改变您解析它的方式。你想找到s并遍历它们,然后在每个你可以使用next_element得到;像这样:doc=Nokogiri::HTML(

  10. ruby-on-rails - 如何在 Rails 3 中禁用 XML 解析 - 2

    我想禁用HTTP参数的自动XML解析。但我发现命令仅适用于Rails2.x,它们都不适用于3.0:config.action_controller.param_parsers.deleteMime::XML(application.rb)ActionController::Base.param_parsers.deleteMime::XMLRails3.0中的等价物是什么? 最佳答案 根据CVE-2013-0156的最新安全公告你可以将它用于Rails3.0。3.1和3.2ActionDispatch::ParamsParser::

随机推荐