用户量比较大或者用户地理位置分布范围很广的项目,通常都会有多个机房。这个时候若是上线springCloud服务的话,咱们但愿一个机房内的服务优先调用同一个机房内的服务,当同一个机房的服务不可用的时候,再去调用其它机房的服务,以达到减小延时的做用。java
eureka提供了region和zone两个概念来进行分区,这两个概念均来自于亚马逊的AWS:spring
如图所示,有一个region:beijing,下面有zone-1和zone-2两个分区,每一个分区内有一个注册中心Eureka Server和一个服务提供者Service。
咱们在zone-1内建立一个Consumer-1服务消费者的话,其会优先调用同一个zone内的Service-1,当Service-1不可用时,才会去调用zone-2内的Service-2。网络
Eureka Server-1:架构
spring: application: name: Server-1 server: port: 30000 eureka: instance: prefer-ip-address: true status-page-url-path: /actuator/info health-check-url-path: /actuator/health hostname: localhost client: register-with-eureka: true fetch-registry: true prefer-same-zone-eureka: true #地区 region: beijing availability-zones: beijing: zone-1,zone-2 service-url: zone-1: http://localhost:30000/eureka/ zone-2: http://localhost:30001/eureka/
Eureka Server-2:app
spring: application: name: Server-2 server: port: 30001 eureka: instance: prefer-ip-address: true status-page-url-path: /actuator/info health-check-url-path: /actuator/health hostname: localhost client: register-with-eureka: true fetch-registry: true prefer-same-zone-eureka: true #地区 region: beijing availability-zones: beijing: zone-2,zone-1 service-url: zone-1: http://localhost:30000/eureka/ zone-2: http://localhost:30001/eureka/
Service-1:测试
测试代码:fetch
@RestController public class HiController { @Value("${zone.name}") private String zoneName; @RequestMapping(value = "/hi", method = RequestMethod.GET) public String hi() { return zoneName; } }
配置文件:url
spring: application: name: service server: port: 30010 eureka: instance: prefer-ip-address: true status-page-url-path: /actuator/info health-check-url-path: /actuator/health metadata-map: zone: zone-1 client: register-with-eureka: true fetch-registry: true prefer-same-zone-eureka: true #地区 region: beijing availability-zones: beijing: zone-1,zone-2 service-url: zone-1: http://localhost:30000/eureka/ zone-2: http://localhost:30001/eureka/ zone.name: zone-1
Service-2:spa
spring: application: name: service server: port: 30011 eureka: instance: prefer-ip-address: true status-page-url-path: /actuator/info health-check-url-path: /actuator/health metadata-map: zone: zone-2 client: register-with-eureka: true fetch-registry: true prefer-same-zone-eureka: true #地区 region: beijing availability-zones: beijing: zone-2,zone-1 service-url: zone-1: http://localhost:30000/eureka/ zone-2: http://localhost:30001/eureka/ zone.name: zone-2
Consumer-1:rest
调用服务代码:
@RestController public class HiController { @Autowired private RestTemplate restTemplate; @RequestMapping(value="/consumer") public String hi() { return restTemplate.getForObject("http://service/hi", String.class); } }
配置文件:
spring: application: name: consumer server: port: 30030 eureka: instance: prefer-ip-address: true status-page-url-path: /actuator/info health-check-url-path: /actuator/health metadata-map: zone: zone-1 client: register-with-eureka: true fetch-registry: true prefer-same-zone-eureka: true #地区 region: beijing availability-zones: beijing: zone-1,zone-2 service-url: zone-1: http://localhost:30000/eureka/ zone-2: http://localhost:30001/eureka/
能够看到Consumer-1优先调用的是同一个zone-1的Service-1,这个时候,不管怎么刷新,调用多少次,都只会调用Service-1,不会调用Service-2.
当咱们把Service-1服务停掉,再调用的话:
才会调用zone-2分区下的Service-2。
整个分区分为两步:
eureka: client: prefer-same-zone-eureka: true #地区 region: beijing availability-zones: beijing: zone-1,zone-2 service-url: zone-1: http://localhost:30000/eureka/ zone-2: http://localhost:30001/eureka/
当一个服务(做为一个eureka client)向注册中心(eureka server)注册的时候,会根据eureka.client下的配置来进行注册。这里咱们主要关心有多个注册中心的状况下,服务会注册到哪一个注册中心,而且和哪一个注册中心来维持心跳检测。
注册中心选择逻辑:
1. 若是prefer-same-zone-eureka为false,按照service-url下的 list取第一个注册中心来注册,并和其维持心跳检测。不会再向list内的其它的注册中心注册和维持心跳。只有在第一个注册失败的状况下,才会依次向其它的注册中心注册,总共重试3次,若是3个service-url都没有注册成功,则注册失败。每隔一个心跳时间,会再次尝试。
2. 若是prefer-same-zone-eureka为true,先经过region取availability-zones内的第一个zone,而后经过这个zone取service-url下的list,并向list内的第一个注册中心进行注册和维持心跳,不会再向list内的其它的注册中心注册和维持心跳。只有在第一个注册失败的状况下,才会依次向其它的注册中心注册,总共重试3次,若是3个service-url都没有注册成功,则注册失败。每隔一个心跳时间,会再次尝试。
因此说,为了保证服务注册到同一个zone的注册中心,必定要注意availability-zones的顺序,必须把同一zone写在前面
eureka: instance: metadata-map: zone: zone-1
服务消费者和服务提供者分别属于哪一个zone,均是经过eureka.instance.metadata-map.zone来断定的。
服务消费者会先经过ribbon去注册中心拉取一份服务提供者的列表,而后经过eureka.instance.metadata-map.zone指定的zone进行过滤,过滤以后若是同一个zone内的服务提供者有多个实例,则会轮流调用。
只有在同一个zone内的全部服务提供者都不可用时,才会调用其它zone内的服务提供者。
eureka.instance.lease-renewal-interval-in-seconds: 30
服务和注册中心的心跳间隔时间,默认为30s
eureka.instance.lease-expiration-duration-in-seconds: 90
服务和注册中心的心跳超时时间,默认为90s
也就是说,当一个服务异常down掉后,90s以后注册中心才会知道这个服务不可用了。在此期间,依旧会把这个服务当成正常服务。ribbon调用仍会把请求转发到这个服务上。为了不这段期间出现没法提供服务的状况,要开启ribbon的重试功能,去进行其它服务提供者的重试。