Eureka的基本架构
- Eureka Server
提供服务注册和发现
- Service Provider
服务提供方
将自身服务注册到Eureka,从而使服务消费方能够找到
- Service Consumer
服务消费方
从Eureka获取注册服务列表,从而能够消费服务
服务消费者通过注册中心获取服务列表,并调用。
Eureka中的元数据:服务的主机名,ip等信息,可以通过eureka server 进行获取,用于服务之间的调用。
org.springframework.cloud.client.discovery.DiscoveryClient;
可以通过这个工具类拿到已经注册到eureka server上的服务元数据
Eureka Server高可用集群
Eureka Server可以通过运行多个实例并相互注册的方式实现高可用部署,Eureka Server实例会彼此增量地同步信息,从而确保所有节点数据一致。事实上,节点之间相互注册是Eureka Server的默认行为。
假如,服务提供者启动只注册到其中一台Eureka Server上,但是你会发现两台Eureka Server上都会有这个服务提供者,这就证明了Eureka Server集群之间会有一个服务同步的过程。这样做是不好的,假如刚好注册的那台Eureka Server 宕机了,这个服务就不可用,所以,在服务启动的时候就要把它注册到Eureka Server 集群的每一台服务器上。
eureka:
client:
service-url:
defaultZone: http://localhost:9000/eureka/,http://localhost:8000/eureka/,...
细节问题
1.在Eureka Server控制台显示服务IP
在服务提供者的yml中,配置eureka.instance.instatnce-id
eureka:
client:
service-url:
defaultZone: http://localhost:9000/eureka/,http://localhost:8000/eureka/
instance:
prefer-ip-address: true #使用ip地址注册
instance-id: ${spring.cloud.client.ip-address}:${server.port} #向注册中心中注册服务id
2.Eureka的服务剔除问题
服务提供者向Eureka Server每隔30s发送一次心跳,告诉它我还活着;如果90s没有发送心跳,就证明这个服务提供者已宕机。在开发阶段时间有点长。
所以在服务的提供者设置心跳间隔,设置续约到期时间。
#配置eureka
eureka:
client:
service-url:
defaultZone: http://localhost:9000/eureka/,http://localhost:8000/eureka/
instance:
prefer-ip-address: true #使用ip地址注册
instance-id: ${spring.cloud.client.ip-address}:${server.port} #向注册中心中注册服务id
lease-renewal-interval-in-seconds: 5 #发送心跳时间
lease-expiration-duration-in-seconds: 10 #续约到期的时间
3.Eureka的自我保护机制
也就是在Eureka控制台出现了一些红色字体
Eureka Server 会统计在15分钟内超过85%的客户端节点都没有正常的心跳,就会开启自我保护机制。此时会出现以下几种情况:
Eureka Server不再从注册列表中移除因为长时间没收到心跳而应该过期的服务。
Eureka Server仍然能够接受新服务的注册和查询请求,但是不会被同步到其它节点上,保证当前节点依然可用。
当网络稳定时,当前Eureka Server新的注册信息会被同步到其它节点中。
因此Eureka Server可以很好的应对因网络故障导致部分节点失联的情况,而不会像ZK那样如果有一半不可用的情况会导致整个集群不可用而变成瘫痪。
eureka:
server:
enable-self-preservation: false #关闭eureka自我保护机制
eviction-interval-timer-in-ms: 4000 #设置剔除服务间隔
Eureka源码解析
SpringBoot中的自动装载
1.ImportSelector
ImportSelector接口是Spring导入外部配置的核心接口,在SpringBoot的自动化配置和@EnableXXX(功能性注解)中起到了决定性的作用。当在@Configuration标注的Class上使用@Import引入了一个ImportSelector实现类后,会把实现类中返回的Class名称都定义为Bean
public interface ImportSelector {
String[] selectImports(AnnotationMetadata var1);
}
DeferredImportSelector接口继承ImportSelector,它和ImportSelector的区别在于装载bean的时机上,DeferredImportSelector需要所有的@Configuration都执行完毕后才会进行装载
public interface DeferredImportSelector extends ImportSelector {
...
}
Eureka Server