该注解用于进行服务发现,其实在新版中该注解已经可以省略。其中,autoRegister属性用于配置是否启用自动注册的功能。作为服务发现实例,主要是注入DiscoveryClient实例,用于服务发现。
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@Import(EnableDiscoveryClientImportSelector.class)
public @interface EnableDiscoveryClient {
boolean autoRegister() default true;
}
在@EnableDiscoveryClient注解上,通过@Import注解引入了EnableDiscoveryClientImportSelector类,通过selectImports()方法,返回了需要注入到容器中的AutoServiceRegistrationConfiguration对象。实现如下:
@Override
public String[] selectImports(AnnotationMetadata metadata) {
String[] imports = super.selectImports(metadata);
AnnotationAttributes attributes = AnnotationAttributes.fromMap(
metadata.getAnnotationAttributes(getAnnotationClass().getName(), true));
boolean autoRegister = attributes.getBoolean("autoRegister");
if (autoRegister) {
List<String> importsList = new ArrayList<>(Arrays.asList(imports));
importsList.add(
"org.springframework.cloud.client.serviceregistry.AutoServiceRegistrationConfiguration");
imports = importsList.toArray(new String[0]);
}
else {
Environment env = getEnvironment();
if (ConfigurableEnvironment.class.isInstance(env)) {
ConfigurableEnvironment configEnv = (ConfigurableEnvironment) env;
LinkedHashMap<String, Object> map = new LinkedHashMap<>();
map.put("spring.cloud.service-registry.auto-registration.enabled", false);
MapPropertySource propertySource = new MapPropertySource(
"springCloudDiscoveryClient", map);
configEnv.getPropertySources().addLast(propertySource);
}
}
return imports;
}
通过上述selectImports()注入了AutoServiceRegistrationConfiguration对象。该类中的代码,已经迁移到了AutoServiceRegistrationAutoConfiguration类,而AutoServiceRegistrationAutoConfiguration类将通过SpringBoot加载META-INF/spring.factories文件中配置类的方法进行加载
@Configuration(proxyBeanMethods = false)
@EnableConfigurationProperties(AutoServiceRegistrationProperties.class)
@ConditionalOnProperty(value = "spring.cloud.service-registry.auto-registration.enabled",
matchIfMissing = true)
public class AutoServiceRegistrationConfiguration {
}
AutoServiceRegistrationAutoConfiguration 配置类,主要是用来判断初始化是否成功,判断依据是:Spring容器中是否已经注入了AutoServiceRegistration对象,该对象是在EurekaClientAutoConfiguration配置类中进行加载的。
@Configuration(proxyBeanMethods = false)
@Import(AutoServiceRegistrationConfiguration.class)
@ConditionalOnProperty(value = "spring.cloud.service-registry.auto-registration.enabled",
matchIfMissing = true)
public class AutoServiceRegistrationAutoConfiguration {
@Autowired(required = false)
private AutoServiceRegistration autoServiceRegistration;
@Autowired
private AutoServiceRegistrationProperties properties;
@PostConstruct
protected void init() {
if (this.autoServiceRegistration == null && this.properties.isFailFast()) {
throw new IllegalStateException("Auto Service Registration has "
+ "been requested, but there is no AutoServiceRegistration bean");
}
}
}
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
org.springframework.cloud.netflix.eureka.config.EurekaClientConfigServerAutoConfiguration,\
org.springframework.cloud.netflix.eureka.config.DiscoveryClientOptionalArgsConfiguration,\
org.springframework.cloud.netflix.eureka.EurekaClientAutoConfiguration,\
org.springframework.cloud.netflix.ribbon.eureka.RibbonEurekaAutoConfiguration,\
org.springframework.cloud.netflix.eureka.EurekaDiscoveryClientConfiguration,\
org.springframework.cloud.netflix.eureka.reactive.EurekaReactiveDiscoveryClientConfiguration,\
org.springframework.cloud.netflix.eureka.loadbalancer.LoadBalancerEurekaAutoConfiguration
org.springframework.cloud.bootstrap.BootstrapConfiguration=\
org.springframework.cloud.netflix.eureka.config.EurekaConfigServerBootstrapConfiguration
在spring-cloud-netflix-eureka-client.jar包下的spring.factories文件中,定义了好多自动配置类,如上所示,包括reactive相关的、负载均衡相关的、服务发现配置和配置中心相关的、还有用于服务发现和服务注册的配置类EurekaClientAutoConfiguration和EurekaDiscoveryClientConfiguration,我们后续就是通过这个两个类来分析Eureka客户端的初始化过程。
该类在spring.factories文件中配置,会通过SpringBoot自动装配机制进行加载。该配置类主要在符合@ConditionalOnXXX条件的情况下,加载了一些实例对象包括了EurekaDiscoveryClient(用于和Eureka服务端进行交互,内部通过EurekaClient实现)、EurekaHealthCheckHandler(实例运行状态处理器)、EurekaClientConfigurationRefresher(ApplicationListener实现类,用于监听RefreshScopeRefreshedEvent事件),最后还有一个Marker,该类实例对象原来用于控制是否加载EurekaClientAutoConfiguration 配置类,现在已经不用了。
该类主要注入Eureka客户端配置对象、服务实例配置对象、EurekaClient对象(实际上是CloudEurekaClient对象)、EurekaRegistration对象、EurekaAutoServiceRegistration对象、EurekaHealthIndicator对象等。
在EurekaClientAutoConfiguration配置类中通过@Bean注解加载EurekaAutoServiceRegistration对象,该对象是一个实现了SmartLifecycle接口的类实例,所以当Spring容器加载所有bean并完成初始化之后,会接着回调该对象的start()方法。
关于SmartLifecycle接口用法,请参考 《Spring SmartLifecycle 在容器所有bean加载和初始化完毕执行》。
//EurekaClientAutoConfiguration配置类
@Bean
@ConditionalOnBean(AutoServiceRegistrationProperties.class)
@ConditionalOnProperty(
value = "spring.cloud.service-registry.auto-registration.enabled",
matchIfMissing = true)
public EurekaAutoServiceRegistration eurekaAutoServiceRegistration(
ApplicationContext context, EurekaServiceRegistry registry,
EurekaRegistration registration) {
return new EurekaAutoServiceRegistration(context, registry, registration);
}
EurekaAutoServiceRegistration对象的start()方法:
//EurekaAutoServiceRegistration类
public class EurekaAutoServiceRegistration implements AutoServiceRegistration,
SmartLifecycle, Ordered, SmartApplicationListener {
//省略其他……
@Override
public void start() {
// 判断协议类型,配置使用的端口号
if (this.port.get() != 0) {
if (this.registration.getNonSecurePort() == 0) {
this.registration.setNonSecurePort(this.port.get());
}
if (this.registration.getSecurePort() == 0 && this.registration.isSecure()) {
this.registration.setSecurePort(this.port.get());
}
}
// 判断当前实例的运行状态,(为什么只判断NonSecurePort端口,不判断SecurePort?)
if (!this.running.get() && this.registration.getNonSecurePort() > 0) {
//注册EurekaRegistration对象,
this.serviceRegistry.register(this.registration);
this.context.publishEvent(new InstanceRegisteredEvent<>(this,
this.registration.getInstanceConfig()));
this.running.set(true);
}
}
}
在start()方法中,主要是调用是调用serviceRegistry.register()方法完成注册,然后发布一个InstanceRegisteredEvent事件,最后修改本地对象的running状态。
在EurekaAutoServiceRegistration对象的start()方法中,主要实现了EurekaRegistration对象(服务实例)注册。
@Override
public void register(EurekaRegistration reg) {
//初始化客户端信息,主要用来初始化cloudEurekaClient==null情况
maybeInitializeClient(reg);
if (log.isInfoEnabled()) {
log.info("Registering application "
+ reg.getApplicationInfoManager().getInfo().getAppName()
+ " with eureka with status "
+ reg.getInstanceConfig().getInitialStatus());
}
reg.getApplicationInfoManager()
.setInstanceStatus(reg.getInstanceConfig().getInitialStatus());
//注册实例健康检查器
reg.getHealthCheckHandler().ifAvailable(healthCheckHandler -> reg
.getEurekaClient().registerHealthCheck(healthCheckHandler));
}
主要在CloudEurekaClient 对象为空的情况下,进行初始化,实际通过调用EurekaRegistration对象的getEurekaClient()方法实现。
private void maybeInitializeClient(EurekaRegistration reg) {
// force initialization of possibly scoped proxies
reg.getApplicationInfoManager().getInfo();
reg.getEurekaClient().getApplications();
}
EurekaRegistration对象的getEurekaClient()方法,主要完成了CloudEurekaClient 对象的初始化(如果不存在的情况下)。
public CloudEurekaClient getEurekaClient() {
if (this.cloudEurekaClient.get() == null) {
try {
this.cloudEurekaClient.compareAndSet(null,
getTargetObject(eurekaClient, CloudEurekaClient.class));
}
catch (Exception e) {
log.error("error getting CloudEurekaClient", e);
}
}
return this.cloudEurekaClient.get();
}
该方法主要用来设置实例的状态,该状态用来表示该实例是否可以接收信息。同时会通知所有的监听器一个状态改变的事件消息StatusChangeEvent。
public synchronized void setInstanceStatus(InstanceStatus status) {
InstanceStatus next = instanceStatusMapper.map(status);
if (next == null) {
return;
}
InstanceStatus prev = instanceInfo.setStatus(next);
if (prev != null) {
for (StatusChangeListener listener : listeners.values()) {
try {
listener.notify(new StatusChangeEvent(prev, next));
} catch (Exception e) {
logger.warn("failed to notify listener: {}", listener.getId(), e);
}
}
}
}
除了EurekaAutoServiceRegistration这条线之外,还有就是实例化EurekaDiscoveryClient对象,在初始化该对象是需要注入EurekaClient对象,而EurekaClient对象就是在配置文件EurekaClientAutoConfiguration配置类中通过@Bean注入的一个对象,实际上使用的是CloudEurekaClient对象。
而CloudEurekaClient又是 DiscoveryClient类的实现类,所以在构建对象时,就会调用DiscoveryClient类的构造函数,所在在前面《DiscoveryClient接口的层级结构》中提到的那个构造函数就会被调用,其实Eureka实例的初始化工作,很多就在这个构造函数中完成了,我们这里主要分析initScheduledTasks()方法中的逻辑。
initScheduledTasks()方法是用来初始化定时任务,其中包括了服务获取和服务注册(心跳、续约)两部分内容,我们这里先学习其中服务获取的相关代码,如下所示:
private void initScheduledTasks() {
//服务获取,会通过REST请求从Eureka服务中获取其他Eureka客户端(服务实例)的信息,形成服务实例清单,缓存到本地。
if (clientConfig.shouldFetchRegistry()) {
// 刷新间隔,默认30s
int registryFetchIntervalSeconds = clientConfig.getRegistryFetchIntervalSeconds();
//失败重试次数,默认10次
int expBackOffBound = clientConfig.getCacheRefreshExecutorExponentialBackOffBound();
//创建线程,用于服务发现和定时更新服务实例信息
cacheRefreshTask = new TimedSupervisorTask(
"cacheRefresh",
scheduler,
cacheRefreshExecutor,
registryFetchIntervalSeconds,
TimeUnit.SECONDS,
expBackOffBound,
new CacheRefreshThread()
);
//执行任务
scheduler.schedule(
cacheRefreshTask,
registryFetchIntervalSeconds, TimeUnit.SECONDS);
}
//……
}
根据上述代码,我们可以知道,服务发现的流程如下:
关于服务发现的具体实现逻辑请参考《Eureka客户端——服务发现》。
initScheduledTasks()方法实现服务续约、注册相关逻辑如下所示:
private void initScheduledTasks() {
//……
//服务注册和续约
if (clientConfig.shouldRegisterWithEureka()) {
//续约时间间隔
int renewalIntervalInSecs = instanceInfo.getLeaseInfo().getRenewalIntervalInSecs();
//失败重试次数
int expBackOffBound = clientConfig.getHeartbeatExecutorExponentialBackOffBound();
logger.info("Starting heartbeat executor: " + "renew interval is: {}", renewalIntervalInSecs);
// 创建心跳服务线程,同时进行服务续约
heartbeatTask = new TimedSupervisorTask(
"heartbeat",
scheduler,
heartbeatExecutor,
renewalIntervalInSecs,
TimeUnit.SECONDS,
expBackOffBound,
new HeartbeatThread()
);
//执行定时任务
scheduler.schedule(
heartbeatTask,
renewalIntervalInSecs, TimeUnit.SECONDS);
// 创建注册线程,主要用于服务注册和节点间的数据同步
instanceInfoReplicator = new InstanceInfoReplicator(
this,
instanceInfo,
clientConfig.getInstanceInfoReplicationIntervalSeconds(),
2); // burstSize
//创建状态监听器,维护响应状态
statusChangeListener = new ApplicationInfoManager.StatusChangeListener() {
@Override
public String getId() {
return "statusChangeListener";
}
@Override
public void notify(StatusChangeEvent statusChangeEvent) {
if (InstanceStatus.DOWN == statusChangeEvent.getStatus() ||
InstanceStatus.DOWN == statusChangeEvent.getPreviousStatus()) {
// log at warn level if DOWN was involved
logger.warn("Saw local status change event {}", statusChangeEvent);
} else {
logger.info("Saw local status change event {}", statusChangeEvent);
}
instanceInfoReplicator.onDemandUpdate();
}
};
//注册监听器到applicationInfoManager实例对象(默认注册,即判断条件默认为true)
if (clientConfig.shouldOnDemandUpdateStatusChange()) {
applicationInfoManager.registerStatusChangeListener(statusChangeListener);
}
//启用服务注册线程
instanceInfoReplicator.start(clientConfig.getInitialInstanceInfoReplicationIntervalSeconds());
} else {
logger.info("Not registering with Eureka server per configuration");
}
}
根据上述代码,我们可以知道,服务发现的流程如下:
boolean renew() {
EurekaHttpResponse<InstanceInfo> httpResponse;
try {
httpResponse = eurekaTransport.registrationClient.sendHeartBeat(instanceInfo.getAppName(), instanceInfo.getId(), instanceInfo, null);
logger.debug(PREFIX + "{} - Heartbeat status: {}", appPathIdentifier, httpResponse.getStatusCode());
if (httpResponse.getStatusCode() == Status.NOT_FOUND.getStatusCode()) {
REREGISTER_COUNTER.increment();
logger.info(PREFIX + "{} - Re-registering apps/{}", appPathIdentifier, instanceInfo.getAppName());
long timestamp = instanceInfo.setIsDirtyWithTime();
boolean success = register();
if (success) {
instanceInfo.unsetIsDirty(timestamp);
}
return success;
}
return httpResponse.getStatusCode() == Status.OK.getStatusCode();
} catch (Throwable e) {
logger.error(PREFIX + "{} - was unable to send heartbeat!", appPathIdentifier, e);
return false;
}
}
在方法中,主要是通过eurekaTransport对象中维护的registrationClient对象,然后调用该对象的sendHeartBeat()方法实现。具体实现请参考《Eureka客户端——服务续约》。
实现服务注册的方法主要是通过InstanceInfoReplicator对象实现,该对象实现了Runnable接口,其中实现的run()方法,即实现了服务注册功能。
public void run() {
try {
//刷新实例信息
discoveryClient.refreshInstanceInfo();
//是否有状态更新过了,有的话获取更新的时间
Long dirtyTimestamp = instanceInfo.isDirtyWithTime();
//有脏数据,要重新注册
if (dirtyTimestamp != null) {
discoveryClient.register();
//设置更新标记为不更新
instanceInfo.unsetIsDirty(dirtyTimestamp);
}
} catch (Throwable t) {
logger.warn("There was a problem with the instance info replicator", t);
} finally {
Future next = scheduler.schedule(this, replicationIntervalSeconds, TimeUnit.SECONDS);
scheduledPeriodicRef.set(next);
}
}
然后,关于服务注册的详细内容,可以参考《Eureka客户端——服务注册》。