微服务实战SpringCloud之Eureka简介

简介

服务发现与治理是微服务体系结构的重要组成之一,在生产环境下由于服务器较多,手动配置很难完成,且一旦变化,如弹性扩容与下线,就会变得很复杂。eureka是Netflix开源的服务发现与治理框架,在Netflix等公司历经实战,拥有极高的可用性。

服务端使用eureka

搭建eureka服务

在springboot项目中,引入eureka-server的依赖group ID 为 org.springframework.cloud , artifact ID 为 spring-cloud-starter-netflix-eureka-server

@SpringBootApplication
@EnableEurekaServer
public class Application {

    public static void main(String[] args) {
        new SpringApplicationBuilder(Application.class).web(true).run(args);
    }

}

高可用性

eureka没有后端存储,但是每个eureka客户端都要向eureka服务端注册自己的实例,并获得其他实例信息,所以,eureka客户端其他实例信息保存在了本地内存中,eureka客户端彼此之间通信不必每次都先调用eureka server端获取实力信息,默认情况下,每30秒客户端向服务端发送一次心跳消息,并更新注册信息。

多个eureka server实例

生产情况下,eureka server是有多个的,eureka server本身也是eureka client,每一个eureka server需要向其他eureka server注册自己,配置示例如下:

peer1的配置

---
spring:
  profiles: peer1
eureka:
  instance:
    hostname: peer1
  client:
    serviceUrl:
      defaultZone: http://peer2/eureka/

peer2的配置

spring:
  profiles: peer2
eureka:
  instance:
    hostname: peer2
  client:
    serviceUrl:
      defaultZone: http://peer1/eureka/

在本地开发时,eureka客户端最好不要注册到eureka服务端,以免其他人在开发时,其他微服务负载到你本机启动的实例中,eureka支持客户端不注册到eureka server。

eureka:
  client:
    registerWithEureka: false
    fetchRegistry: false
    serviceUrl:
      defaultZone: http://peer1/eureka/

eureka server的CSRF支持

eureka 需要通过spring security来防范CSRF攻击,引入spring-security依赖spring-boot-starter-security,再加上Bean配置。

@EnableWebSecurity
class WebSecurityConfig extends WebSecurityConfigurerAdapter {

    @Override
    protected void configure(HttpSecurity http) throws Exception {
        http.csrf().ignoringAntMatchers("/eureka/**");
        super.configure(http);
    }
}

eureka自我保护模式

eureka客户端向服务端注册,并默认情况下每30秒发送一次心跳消息以向eureka server说明自己的实例运行正常,也称之为服务续约。当某个实例在连续3次未能注册心跳消息时,eureka会将该实例标注为“DOWN”,如果在短时间内有超过15%的服务down掉,eureka认为可能是网络原因,其他服务未能向自己注册,而并不是服务已经down掉。因此,eureka会默认启动自我保护模式,不会下线这些实例。

但是当我们本地开发时,由于服务本身就很少,很容易就能达到阈值,因此有必要关掉自我保护模式,关闭的配置如下:eureka.enableSelfPreservation=false

客户端使用eureka-client

客户端引入eureka-client

项目中引入eureka-client,使用maven或gradle引入eureka-client依赖。group ID 是 org.springframework.cloud , artifact ID 为 spring-cloud-starter-netflix-eureka-client.

注册到eureka服务端

当一个客户端注册到eureka服务端时,它会将自身的IP、端口、主页等其他信息提供给eureka服务端。eureka服务端接收客户端的心跳消息,当心跳故障超过一定时间(可配置),通常会将该实例从已注册实例中移除。

以下示例显示了最小的Eureka客户端应用程序:

@SpringBootApplication
@RestController
public class Application {

    @RequestMapping("/")
    public String home() {
        return "Hello world";
    }

    public static void main(String[] args) {
        new SpringApplicationBuilder(Application.class).web(true).run(args);
    }

}

客户端需增加以下配置:这里以yml文件为例

eureka:
  client:
    serviceUrl:
      defaultZone: http://localhost:8761/eureka/

在上面的示例中,defaultZone是一个魔法值,上面的值是它默认的URL。eureka客户端的配置通过eureka.instance.*来指定,spring.application.name要配置。

如果要禁用eureka,可以通过设置eureka.client.enabledfalse

使用eureka server进行服务认证

如果其中一个eureka.client.serviceUrl.defaultZoneURL中嵌入了认证,则会自动将HTTP基本身份验证添加到您的eureka客户端(如下所示:) http://user:password@localhost:8761/eureka。对于更复杂的需求,可以创建一个@Bean类型DiscoveryClientOptionalArgs并将ClientFilter实例注入其中,所有这些实例都应用于从客户端到服务器的调用。

状态页面和健康指标页面

eureka的状态页面和健康指标页面分别默认为/info/health,如果使用非默认上下文路径或servlet路径(例如server.servletPath=/custom),则需要更改这些,即使对于Actuator应用程序也是如此。以下示例显示了这两个设置的默认值:

eureka:
  instance:
    statusPageUrlPath: ${server.servletPath}/info
    healthCheckUrlPath: ${server.servletPath}/health

eureka的健康检查

默认情况下,eureka是通过客户端发送的心跳消息来判断客户端是否处于运行状态。通常,eureka客户端不会根据Spring Boot Actuator来传播客户端的当前运行状况检查状态。因此在成功注册后,客户端在eureka那里始终处于“UP”状态。如果想将客户端状态传播至eureka,那么可以启用eureka的健康检查,这样其他应用程序都不会向“UP”以外的状态下的应用程序发送流量。

application.yml.

eureka:
  client:
    healthcheck:
      enabled: true

注意,这里是application.yml,不是bootstrap.yml,如果在bootstrap.yml中配置了此配置,可能会发生一些意向不到的情况,如eureka那里显示应用程序为“UNKNOWN”,因为bootstrap.yml的配置优先级较高。

如果想要更为复杂的健康检查,就要实现自己的健康检查类,com.netflix.appinfo.HealthCheckHandler

使用EurekaClient

在注册到eureka服务后,微服务之间调用通过负载均衡调用时,可以通过如下方式来获取服务的URL:

@Autowired
private EurekaClient discoveryClient;

public String serviceUrl() {
    InstanceInfo instance = discoveryClient.getNextServerFromEureka("STORES", false);
    return instance.getHomePageUrl();
}

注意:不要在@PostConstruct方法或@Scheduled方法中使用(或者ApplicationContext可能尚未启动的任何地方)。

不使用jersey

默认情况下,eurekaClient是通过Jersey进行http通信的,如果不想使用Jersey,可以通过下面的方式去除Jersey的依赖,以及使用restTemplate进行http通信。

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
    <exclusions>
        <exclusion>
            <groupId>com.sun.jersey</groupId>
            <artifactId>jersey-client</artifactId>
        </exclusion>
        <exclusion>
            <groupId>com.sun.jersey</groupId>
            <artifactId>jersey-core</artifactId>
        </exclusion>
        <exclusion>
            <groupId>com.sun.jersey.contribs</groupId>
            <artifactId>jersey-apache-client4</artifactId>
        </exclusion>
    </exclusions>
</dependency>
@EnableDiscoveryClient
@SpringBootApplication
public class CartApplication {

    public static void main(String[] args) {
        SpringApplication.run(CartApplication.class, args);
    }
    
    @Bean
    @LoadBalanced
    public RestTemplate restTemplate(){
        return new RestTemplate();
    }
}

为什么注册服务这么慢

在使用eureka的时候吗,有时候可能会发现明明eureka客户端已经注册到eureka服务上了,可是其他的客户端却找不到该服务。原因是每个eureka客户端在本地都会有一份eureka实例的缓存,默认情况下,每30秒客户端会向服务端发送一次心跳检查,并更新服务实例信息。如果想要加快速度,可以通过eureka.instance.leaseRenewalIntervalInSeconds 来指定时间。

zone区域

在生产环境下,我们可能会将eureka客户端及服务端部署到不同的区域,如华北,华南区域,相同区域的服务之间相互调用网络延迟会相对较少一些。eureka中可以通过配置,指定实例的zone区域。

如:

一区

eureka.instance.metadataMap.zone = zone1
eureka.client.preferSameZoneEureka = true

二区

eureka.instance.metadataMap.zone = zone2
eureka.client.preferSameZoneEureka = true
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,921评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,635评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,393评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,836评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,833评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,685评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,043评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,694评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,671评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,670评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,779评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,424评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,027评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,984评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,214评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,108评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,517评论 2 343

推荐阅读更多精彩内容