Spring Cloud Hystrix(七)
Spring Cloud Hystrix
本章节的代码:https://github.com/wicksonZhang/Spring-Cloud
我们只需要聚焦在如下服务当中:
1. 基本概念
1.1. Hystrix
是什么
Hystrix
是Netflix
开源的一个用于构建分布式系统中的延迟和容错的库。主要核心功能有 服务降级、服务熔断、服务限流。- 服务降级:当依赖服务不可用时,返回一个备用的默认值或者执行备选方案,确保系统的基本功能仍然可以使用。
- 服务熔断:服务熔断是一种机制,用于防止故障在分布式系统中蔓延。类似于家庭中的保险丝,服务熔断器会在依赖服务出现故障时进行打开,停止对该服务的调用,避免持续的失败请求导致系统崩溃。
- 服务限流:服务限流是一种控制系统流量的方法,用于防止系统被过多请求压垮。通过限制对某个服务的并发请求或单位时间内的请求数量,可以避免系统因过载而崩溃或变得不稳定。
1.2. Hystrix
优缺点
优点
- 容错性强: Hystrix 提供了服务降级、熔断和限流等机制,可以防止故障在整个系统中扩散,保证系统的稳定性和可用性。
- 提高系统可靠性: 它能够在依赖服务不可用时提供备用方案,确保系统的基本功能可以继续运行。
- 监控和度量: Hystrix 提供了丰富的监控功能,可以实时监控依赖服务的调用情况和性能指标,帮助进行故障排查和性能优化。
缺点
- 目前
Hystrix
已经停止更新了。 - 复杂性增加: 在项目中引入 Hystrix 可能会增加代码复杂性,需要对服务调用进行额外的封装和配置。
1.3. Hystrix
应用场景
在一个微服务架构中,服务之间存在着依赖关系。例如,一个电子商务平台的订单服务依赖于库存服务和支付服务。如果库存服务出现了延迟或者不可用,订单服务可能因为等待库存服务的响应而出现延迟,甚至导致服务不可用。在这种情况下,我们就可以采用 Hystrix
进行处理。
电商平台的库存服务:
- 服务降级: 当库存服务不可用时,订单服务可以返回一个预先设定的默认库存量,确保用户可以继续下单。
- 服务熔断: 如果库存服务连续出现错误响应或超时,Hystrix 可以打开断路器,暂时停止对库存服务的调用,防止因大量失败请求导致订单系统崩溃。
- 服务限流: 控制对库存服务的访问速率,防止订单系统因过多的库存查询请求而受到压力,保持系统稳定性。
在线支付系统的支付服务:
- 服务降级: 当支付服务不可用时,可以返回一个预定义的错误码或信息,告知用户支付暂时不可用,而不是让用户无限等待或看到错误页面。
- 服务熔断: 在支付服务故障频发时,Hystrix 可以打开断路器,阻止进一步的支付请求,避免因支付服务的连续失败导致整个系统不可用。
- 服务限流: 控制支付请求的速率,避免系统因支付请求过多而导致支付服务不稳定或响应时间增加。
2. Spring Cloud Hystrix 具体实现
2.1. 服务降级
实现需求
- 我们还是基于 支付微服务和订单服务的例子进行模拟
实现思路
- 注册中心我们还是延用 Eureka 的集群版。
- Step-1:创建订单服务
08-spring-cloud-hystrix-order-8100
- Step-2:创建支付服务
08-spring-cloud-hystrix-payment-8000
代码结构
2.1.1. 创建支付服务
创建支付服务:
08-spring-cloud-hystrix-payment-8000
实现步骤
- Step-1:导入
pom.xml
依赖 - Step-2:修改
application.properties
文件 - Step-3:创建主启动类
- Step-4:编写业务类
- Step-5:编写控制类
- Step-1:导入
Step-1:导入
pom.xml
依赖1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19<dependencies>
<!-- 引入公共依赖包 -->
<dependency>
<groupId>cn.wickson.cloud</groupId>
<artifactId>01-spring-cloud-common</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
<!-- 服务注册中心的客户端端 eureka-client -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
<!-- 熔断限流 hystrix -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-hystrix</artifactId>
</dependency>Step-2:修改
application.properties
文件1
2
3
4
5
6
7
8
9
10
11
12
13
14# 服务端口
server.port=8000
# 应用名称
spring.application.name=spring-cloud-hystrix-payment
# 是否向注册中心注册自己
eureka.client.register-with-eureka=true
# 表示自己就是注册中心,职责是维护服务实例,并不需要去检索服务
eureka.client.fetch-registry=true
# 设置与eureka server交互的地址查询服务和注册服务都需要依赖这个地址
eureka.client.serviceUrl.defaultZone=http://eureka3300.com:3300/eureka,http://eureka3400.com:3400/eureka
# 设置Eureka服务实例的唯一标识为 spring-cloud-cluster-eureka-payment:3600
eureka.instance.instance-id=spring-cloud-hystrix-payment:8000
# 设置Eureka客户端是否偏好使用IP地址注册到Eureka服务器,而不是使用主机名
eureka.instance.prefer-ip-address=trueStep-3:创建主启动类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15/**
* hystrix 启动类
*
* @author ZhangZiHeng
* @date 2024-01-08
*/
public class SpringCloudHystrixPaymentApplication {
public static void main(String[] args) {
SpringApplication.run(SpringCloudHystrixPaymentApplication.class, args);
}
}Step-4:编写业务类
PaymentServiceImpl.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37/**
* 支付服务-应用服务实现类
*
* @author ZhangZiHeng
* @date 2024-01-08
*/
public class PaymentServiceImpl implements IPaymentService {
/**
* 成功
*
* @return String
*/
public ResultUtil paymentBySuccess() {
return ResultUtil.success("ThreadPool:" + Thread.currentThread().getName() + ", payment service success");
}
/**
* 连接超时
*
* @return String
*/
public ResultUtil paymentByTimeOut() {
int timeNumber = 3;
try {
// 模拟网络延迟
TimeUnit.SECONDS.sleep(timeNumber);
} catch (Exception exception) {
exception.printStackTrace();
}
return ResultUtil.success("ThreadPool:" + Thread.currentThread().getName() + ", payment service timeout:" + timeNumber);
}
}Step-5:编写控制类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29/**
* 支付微服务-控制类
*
* @author ZhangZiHeng
* @date 2024-01-08
*/
public class PaymentController {
private IPaymentService paymentService;
private Integer serverPort;
public ResultUtil paymentSuccess() {
return paymentService.paymentBySuccess();
}
public ResultUtil paymentTimeOut() {
return paymentService.paymentByTimeOut();
}
}
2.1.2. 创建订单服务
订单服务会主动进行调用支付服务,所以我们将
服务降级、服务熔断、服务隔离
写在订单服务中。两端都可以写,我们目前写在调用方(订单服务)中。
创建订单服务:
08-spring-cloud-hystrix-order-8100
实现步骤
- Step-1:导入
pom.xml
依赖 - Step-2:修改
application.properties
文件 - Step-3:创建主启动类
- Step-4:编写业务类
- Step-5:编写控制类
- Step-1:导入
Step-1:导入
pom.xml
依赖1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26<dependencies>
<!-- 公共依赖服务 -->
<dependency>
<groupId>cn.wickson.cloud</groupId>
<artifactId>01-spring-cloud-common</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
<!-- 服务注册中心的客户端端 eureka-client -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
<!-- 服务调用依赖包:OpenFeign -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
<!-- 熔断限流 hystrix -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-hystrix</artifactId>
</dependency>
</dependencies>Step-2:修改
application.properties
文件1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32# 服务端口
server.port=8100
# 应用名称
spring.application.name=spring-cloud-hystrix-order
# ============================= Eureka =================================
# 是否向注册中心注册自己
eureka.client.register-with-eureka=true
# 表示自己就是注册中心,职责是维护服务实例,并不需要去检索服务
eureka.client.fetch-registry=true
# 设置与eureka server交互的地址查询服务和注册服务都需要依赖这个地址
eureka.client.serviceUrl.defaultZone=http://eureka3300.com:3300/eureka,http://eureka3400.com:3400/eureka
# 设置Eureka服务实例的唯一标识为 spring-cloud-cluster-eureka-payment:3600
eureka.instance.instance-id=spring-cloud-hystrix-order:8100
# 设置Eureka客户端是否偏好使用IP地址注册到Eureka服务器,而不是使用主机名
eureka.instance.prefer-ip-address=true
# ============================= Eureka =================================
# ============================= open Feign 配置start =============================
# 建立连接所有的时间,适用于网络状况正常的情况,两端建立所花费时间
feign.client.config.default.connect-timeout=5000
# 建立连接后服务端读取可用资源所有的时间,默认是 1S
feign.client.config.default.read-timeout=5000
# ============================= open Feign 配置end =============================
# ============================= hystrix =================================
# 配置 hystrix 连接超时
feign.hystrix.enabled=true
# https://blog.csdn.net/tszxlzc/article/details/106625387
# 目前设置为5 S,默认是 1S(default_executionTimeoutInMilliseconds = 1000)
hystrix.command.default.execution.isolation.thread.timeoutInMilliseconds=5000
# ============================= hystrix =================================Step-3:创建主启动类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17/**
* 订单服务-启动类
*
* @author ZhangZiHeng
* @date 2024-01-08
*/
// 启用 Hystrix 断路器
// 启用 Eureka 客户端
// 启用 Feign 客户端
public class SpringCloudHystrixOrderApplication {
public static void main(String[] args) {
SpringApplication.run(SpringCloudHystrixOrderApplication.class, args);
}
}Step-4:OpenFeign 调用类
IPaymentHystrixService.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29/**
* Feign远程调用接口支付接口
*
* @author ZhangZiHeng
* @date 2024-01-09
*/
// value 声明一个 Feign 客户端,value 指定了要调用的服务名为 "SPRING-CLOUD-HYSTRIX-PAYMENT",
// fallback 指定了当调用失败时的降级处理类为 PaymentFallbackService.class
public interface IPaymentHystrixService {
/**
* 成功
*
* @return String
*/
public ResultUtil paymentBySuccess();
/**
* 连接超时
*
* @return String
*/
public ResultUtil paymentByTimeOut();
}PaymentFallbackService.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
public class PaymentFallbackService implements IPaymentHystrixService {
public ResultUtil paymentBySuccess() {
return ResultUtil.failure(ResultCodeEnum.SYSTEM_ERROR);
}
public ResultUtil paymentByTimeOut() {
return ResultUtil.failure(ResultCodeEnum.SYSTEM_ERROR);
}
}Step-5:编写控制类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26/**
* 订单服务-控制类
*
* @author ZhangZiHeng
* @date 2024-01-08
*/
public class OrderController {
private IPaymentHystrixService paymentHystrixService;
public ResultUtil getPaymentSuccess() {
return paymentHystrixService.paymentBySuccess();
}
public ResultUtil getPaymentTimeOut() {
return paymentHystrixService.paymentByTimeOut();
}
}注意:我们上面的操作直接做的全局限流操作,其实可以对单个方法进行限流。
1
2
3
4
5
6
7
8
9
10
11
public ResultUtil getPaymentTimeOut() {
return paymentHystrixService.paymentByTimeOut();
}
public ResultUtil paymentTimeOutFallbackMethod(){
return ResultUtil.failure(ResultCodeEnum.SYSTEM_ERROR);
}
2.1.3. 单元测试
正常情况调用
我们将支付服务服务进行关闭,然后调用
2.2. 服务熔断
服务熔断:服务熔断是一种机制,用于防止故障在分布式系统中蔓延。类似于家庭中的保险丝,服务熔断器会在依赖服务出现故障时进行打开,停止对该服务的调用,避免持续的失败请求导致系统崩溃。
实现需求
- 我们还是基于 支付微服务和订单服务的例子进行模拟
实现思路
- 注册中心我们还是延用 Eureka 的集群版。
- Step-1:修改订单服务
08-spring-cloud-hystrix-order-8100
- Step-2:修改支付服务
08-spring-cloud-hystrix-payment-8000
2.2.1. 修改订单服务
Step-1:修改订单服务
08-spring-cloud-hystrix-order-8100
- 新增控制类
1
2
3
4
public ResultUtil getPaymentCircuitBreaker( { Long id)
return paymentHystrixService.paymentCircuitBreaker(id);
}- 新增 Feign 调用
1
2
3
4
5
6
7/**
* 服务熔断
*
* @return
*/
public ResultUtil paymentCircuitBreaker(; Long id)
2.2.2. 修改支付服务
Step-2:修改支付服务
08-spring-cloud-hystrix-payment-8000
- 新增控制类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
public ResultUtil paymentCircuitBreaker( { Long id)
return paymentService.paymentCircuitBreaker(id);
}
public ResultUtil paymentCircuitBreakerFallback( { Long id)
return ResultUtil.failure(ResultCodeEnum.SYSTEM_ERROR, "Fallback: Payment Timeout. Please try again later.");
}
- 新增业务类
1
2
3
4
5
6
7
8
public ResultUtil paymentCircuitBreaker(final Long id) {
if (id < 0) {
throw UserOperationException.getInstance(ResultCodeEnum.PARAM_IS_INVALID);
}
String serialNumber = IdUtil.simpleUUID();
return ResultUtil.success("ThreadPool:" + Thread.currentThread().getName() + ", serialNumber:" + serialNumber);
}
2.2.3. 单元测试
2.2.4. 熔断的状态
状态 | 含义 |
---|---|
熔断打开 OPEN | 请求到达后,不再执行业务逻辑,内部设置时钟,值为平均故障处理时间 熔断打开状态持续到该时钟设定值后,进入熔断半开状态 |
熔断关闭 CLOSED | 熔断关闭,不对微服务进行熔断 |
熔断半开 HALF-OPEN | 部分请求执行业务逻辑,尝试恢复微服务,如果请求成功且符合规则,则关闭熔断 |
3. Hystrix
服务监控
Hystrix
监控中心:Hystrix
提供的一套可视化系统Hystrix-Dashboard
,可以非常友好的看到当前环境中服务运行的状态。实现思路
- Step-1:创建服务监控
08-spring-cloud-hystrix-dashboard-8200
- Step-2:导入
pom.xml
依赖 - Step-3:修改
application.properties
文件 - Step-4:创建主启动类
- Step-1:创建服务监控
代码截图
3.1. 创建 DashBoard
Step-1:创建服务监控
08-spring-cloud-hystrix-dashboard-8200
Step-2:导入
pom.xml
依赖1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20<dependencies>
<!-- 引入公共依赖包 -->
<dependency>
<groupId>cn.wickson.cloud</groupId>
<artifactId>01-spring-cloud-common</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
<!-- 服务注册中心的客户端端 eureka-client -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-hystrix-dashboard</artifactId>
</dependency>
<!-- 熔断限流 hystrix -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-hystrix</artifactId>
</dependency>
</dependencies>Step-3:修改
application.properties
文件1
2
3
4# 服务端口
server.port=8200
# 应用名称
spring.application.name=spring-cloud-hystrix-dashboardStep-4:创建主启动类
1
2
3
4
5
6
7
8
9
public class SpringCloudHystrixDashboardApplication {
public static void main(String[] args) {
SpringApplication.run(SpringCloudHystrixDashboardApplication.class, args);
}
}-
3.2. 启动支付服务
- 在主启动类中添加如下代码
1 | /** |
访问:http://localhost:8000/payment/hystrix/paymentCircuitBreaker/1 制造部分流量
访问:http://localhost:8200/hystrix
3.3. DashBoard
图解
- 实心圆
- 它通过颜色的变化代表了实例的健康程度,它的健康度从绿色<黄色<橙色<红色递减。
- 该实心圆除了颜色的变化之外,它的大小也会根据实例的请求流量发生变化,流量越大该实心圆就越大。所以通过该实心圆的展示,就可以在大量的实例中快速的发现故障实例和高压力实例