Spring Cloud入门教程(四):微服务容错保护(Hystrix)

上一篇:《Spring Cloud入门教程(三):声明式服务调用(Feign)》

本人和同事撰写的《Spring Cloud微服务架构开发实战》一书也在京东、当当等书店上架,大家可以点击这里前往购买,多谢大家支持和捧场!


我们在实践微服务架构时,通常会将业务拆分成一个个微服务,微服务之间通过网络进行通信,进行互相调用,造成了微服务之间存在依赖关系。我们知道由于网络原因或者自身的原因,服务并不能保证服务的100%可用,如果单个服务出现问题,调用这个服务就会出现网络延迟甚至调用失败,而调用失败又会造成用户刷新页面并再次尝试调用,再加上其它服务调用,从而增加了服务器的负载,导致服务瘫痪,最终甚至会导致整个服务“雪崩”。

Netflix为解决这个问题根据断路器模式创建了一个名为Hystrix的库。“断路器”本身是一种开关装置,当某个服务单元发生故障之后,通过断路器的故障监控(类似熔断保险丝),向调用方返回一个符合预期的、可处理的备选响应(FallBack),而不是长时间的等待或者抛出调用方无法处理的异常,这样就保证了服务调用方的线程不会被长时间、不必要地占用,从而避免了故障在分布式系统中的蔓延,乃至雪崩。

HystrixFallback

当然,在请求失败频率较低的情况下,Hystrix还是会直接把故障返回给客户端。只有当失败次数达到阈值(默认在20秒内失败5次)时,断路器打开并且不进行后续通信,而是直接返回备选(FallBack)响应。

1. 示例代码

对于上一篇我们提到了Feign默认是整合了Ribbon和Hystrix这两个框架,所以代码我们在上一篇的基础上进行修改。

首先,需要说明的是实现微服务容错保护的是服务消费方,也就是Mall工程。那接下来我们代码的更改也主要都是对Mall工程进行修改,其它工程的代码将保持不变。

1.1 增加对Hystrix的依赖

在pom文件中增加如下代码:

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-hystrix</artifactId>
</dependency>     

1.2 开启Hystrix的支持

修改Application类:

/**
 * TwoStepsFromJava Cloud -- Mall Web Project
 *
 * @author CD826(CD826Dong@gmail.com)
 * @since 1.0.0
 */
@EnableCircuitBreaker
@EnableFeignClients
@EnableDiscoveryClient
@SpringBootApplication
public class Application {

    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }

}

这里增加了@EnableCircuitBreaker注解,也就是启用断路器支持。

1.3 实现ProductService的FallBack

这里我们新增了一个实现了,并实现了ProductService接口:

/**
 * Product Service Fallback
 *
 * @author CD826(CD826Dong@gmail.com)
 * @since 1.0.0
 */
@Component
public class ProductServiceFallback implements ProductService {
    @Override
    public List<Product> findAll() {
        return Collections.emptyList();
    }

    @Override
    public Product loadByItemCode(@PathVariable("itemCode") String itemCode) {
        return new Product("error", "未知", "TwoStepsFromJava-Fallback", 0);
    }
}

对于findAll方法,我们直接返回了一个空白集合(当前,你可以在这里定义任何你想返回给用户显示的信息)。loadByItemCode方法则返回了一个固定产品信息。

1.3 让ProductService具有容错能力

这个很简单,我们只需要在注解中进行配置即可,如下:

/**
 * Product Service
 *
 * @author CD826(CD826Dong@gmail.com)
 * @since 1.0.0
 */
@FeignClient(name = "PRODUCT-SERVICE", fallback = ProductServiceFallback.class)
public interface ProductService {

    @RequestMapping(value = "/products", method = RequestMethod.GET)
    List<Product> findAll();

    @RequestMapping(value = "/products/{itemCode}", method = RequestMethod.GET)
    Product loadByItemCode(@PathVariable("itemCode") String itemCode);

}

这个仅仅是在@FeignClient注解中增加了fallback的配置,并设置其值为我们刚刚新建的类:ProductServiceFallback

1.4 让Feign启用Hystrix

这个需要在application.properties中增加下面一个配置:

feign.hystrix.enabled=true

1.5 微服务容错测试

  1. 首先,我们启动Eureka Server、两个PRODUCT-SERVICEmall-web,启动后在Eureka Server的监控界面可以看到注册的服务如下:
Hystrix-010
  1. 我们在浏览器中访问:http://localhost:8080/products/item-3,我们可以看到如下界面:
Hystrix-020

这说明我们的服务调用是成功的。

  1. 我们停掉端口为2100PRODUCT-SERVICE服务,这时候在Eureka Server的监控界面可以看到注册的服务如下:
Hystrix-030

说明,PRODUCT-SERVICE的其中一个微服务已经宕机,此时我们再访问上面的地址依然可以得到正确的响应。

  1. 我们继续停掉端口为2200PRODUCT-SERVICE服务,这时候在Eureka Server的监控界面可以看到注册的服务如下:
Hystrix-040

说明,PRODUCT-SERVICE服务全部已经宕机,此时我们再访问上面的地址,将会看到如下界面:

Hystrix-050

可以看到FallBack已经启作用,当全部PRODUCT-SERVICE不起作用时,mall-web中的ProductService进入了回退处理。

1.6 在不使用Feign时如何使用Hystrix

其实Hystrix提供了两个对象来支持回退处理:HystrixCommandHystrixObservableCommand,其中后者是用在依赖的服务返回多个操作结果的时候,这里我们只演示一下HystrixCommand的使用,对于后者可自行尝试,或者查看官方文档

这里假设我们有一个UserController其中findById方法需要进行容错保护,那么代码如下:

@HystrixCommand(fallbackMethod="findByIdFallback")
@RequestMapping(value = "/users/{id}", method = RequestMethod.GET)
public User findById(@PathVariable Long id) {
    return this.restTemplate.getForObject("http://USER-SERVICE/"+ id, User.class);
}

public User findByIdFallback(Long id) {
    User user = new User();
    user.setId(-1);
    user.setName("-default-");
    return user;
}   

可以看出,对于FallBack方法只需要与原方法具有相同的参数及返回值即可,也就是说函数签名要相同。当然@HystrixCommand注解还有很多属性可以自定义,大家可以自行进行尝试。

2. Hystrix容错原理简析

  • 请求封装: 不论是HystrixCommand还是HystrixObervableCommand从类命名上可以看到Hystrix其实是使用了"命令模式",通过命令模式实现对服务调用操作的封装,将每个命令在一个独立线程中进行执行;
  • 跳闸机制: 当某个服务的错误率超过一定阀值时(默认在20秒内失败5次), Hystrix可以自动或手动进行服务跳闸,停止向该服务请求一段时间;
  • 资源隔离: Hystrix为每一个服务依赖都维护了一个小型线程池,如果该线程池已满,那么发往该服务的请求就会立即被拒绝,而不是排队等候,从而加速服务失败的判定;
  • 服务监控: Hystrix可以近乎实时地监控运行指标和配置的变化,例如对请求的成功、失败、超时以及拒绝等;
  • 回退机制: 当请求失败、超时、被拒绝、或当断路器打开时,执行相应的回退逻辑;
  • 自我修复: 当断路器打开一段时间后,Hystrix会进入"半开"状态,断路器会允许一个请求尝试对服务进行请求,如果该服务可以调用成功,则关闭断路器,否则将继续保持断路器打开。

3. Hystrix监控

Hystrix除了实现服务容错之外,还提供了对服务请求的监控:每秒执行的请求数、成功数等。开启Hystrix的监控非常简单,一个是添加spring-cloud-starter-hystrix,这个在之前的示例中以及添加。二是添加spring-boot-starter-actuator,能够让/hystrix-stream端点可以获取到Hystrix的监控数据。

启动服务后,我们在浏览器中输入: http://locaohost:8080/hystrix.stream就会看到下面的界面:

Hystrix-080

可以看到页面会重复输出一些统计数据(注: 你要先尝试访问一下所提供的服务才会有这些数据输出)。至于这些数据到底是什么我在这里就不一一解析了,幸好Hystrix还为我们提供了一个可视化界面来查看这些数据。

3.1 Hystrix Dashboard

当然你可以为Hystrix Dashboard新建一个工程(最好也是),我这里为了简化,直接将Dashboard集成到mall-web工程中。

首先,我们在pom.xml中增加下面的依赖:

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-hystrix-dashboard</artifactId>
</dependency>

然后,在主启动类中增加@EnableHystrixDashboard注解,开启Hystrix Dashboard服务,如:

@EnableHystrixDashboard
@EnableCircuitBreaker
@EnableFeignClients
@EnableDiscoveryClient
@SpringBootApplication
public class Application {

    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }

}

最后,重启mall-web工程即可。

我们在浏览器中输入: http://localhost:8080/hystrix,可以看到如下界面:

Hystrix-060

说明Dashboard已经启动成功。然后在界面中输入之前的地址: http://locaohost:8080/hystrix.stream,然后点击[Monitor Stream]就可以看到统计报表页面:

Hystrix-070

从截图中可以看到有两个服务接口的统计数据:findAllloadByItemCode

每个方法的统计信息中包含两个重要的图形信息:一个实心圆和一个曲线。

  • 实心圆:包含两个含义。颜色表示实例的健康程度,健康程度从绿色、黄色、橙色、红色递减。大小则根据请求流量的大小发生变化,流量越大则实心圆越大,反之则小。
  • 曲线: 统计了2分钟内的请求流量的变化,通过该曲线可以对流量进行上升和下降的趋势分析。

对于界面中的数字,其表示的意义如下:

Hystrix-090

在Dashboard首页时,我们知道Hystrix Dashboard支持三种监控方式:

  • 默认集群监控: 通过http://turbine-hostname:port/turbine.stream
    开启,实现对默认集群的监控;
  • 指定集群监控: 通过http://turbine-hostname:port/turbine.stream?cluster=[clusterName]开启,实现对指定clusterName集群的监控;
  • 单机应用监控: 通过http://hystrix-app:port/hystrix.stream开启,实现对某个服务实例的监控。

在上面的示例中我们演示的第三种方式,至于如何继承Turbine实现对集群的监控,我们将在后续篇幅中进行讲解。

你可以到这里下载本篇的代码。

下一篇:《Spring Cloud入门教程(五):API服务网关(Zuul) 上》

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

推荐阅读更多精彩内容