熔断器(Hystrix,Resilience4j)

news2025/5/24 11:59:39

熔断器

核心原理​

熔断器通过监控服务调用失败率,在达到阈值时自动切断请求,进入熔断状态(类似电路保险丝)。其核心流程为:
关闭状态(Closed)​​:正常处理请求,统计失败率。
​​打开状态(Open)​​:触发熔断,直接拒绝请求,调用降级逻辑。
​​半开状态(Half-Open)​​:允许少量请求尝试恢复,若成功则关闭熔断。

状态流转

  1. 关闭状态(Closed)​​
    ​​行为​​:熔断器关闭,所有请求正常通过。
    ​​触发条件​​:
    初始默认状态。
    从半开状态恢复成功后切换至此状态。
    ​​核心规则​​:
    统计时间窗口内的请求失败率。
    当失败率超过阈值(如 50%)时,熔断器跳转到 ​​打开状态​​。
# Resilience4j 配置(失败率 ≥ 50% 触发熔断)
resilience4j.circuitbreaker:
  instances:
    serviceA:
      failureRateThreshold: 50
      minimumNumberOfCalls: 5  # 最少需要5次调用才计算失败率
      slidingWindowSize: 10     # 统计最近10次调用
  1. 打开状态(Open)​
    行为​​:熔断器打开,所有请求被快速拒绝,直接触发降级逻辑。
    ​​触发条件​​:
    关闭状态下失败率超过阈值。
    ​​核心规则​​:
    熔断器保持打开状态一段时间(如 10秒)。
    超时后自动进入 ​​半开状态​​。
  2. 半开状态(Half-Open)​
    行为​​:允许少量探测请求通过,若成功则恢复服务,否则重新熔断。
    ​​触发条件​​:
    熔断器从打开状态等待指定时间后自动切换。
    ​​核心规则​​:
    允许有限数量的请求(如 3次)试探性调用。
    若成功率达标(如 80%),则关闭熔断器。
    若失败率仍高,则重新进入 ​​打开状态​​。
# Resilience4j 半开状态配置
resilience4j.circuitbreaker:
  instances:
    serviceA:
      permittedNumberOfCallsInHalfOpenState: 3  # 允许3次探测请求
      failureRateThreshold: 20                  # 半开状态下失败率≥20%则重新熔断
      waitDurationInOpenState: 10s              # 打开状态持续10秒后进入半开

Hystrix使用

依赖

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-hystrix</artifactId>
    <version>2.2.10.RELEASE</version> <!-- 对应 Spring Cloud Hoxton.SR12 -->
</dependency>

启用 Hystrix​

@SpringBootApplication
@EnableHystrix          // 启用 Hystrix
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

全局默认配置

hystrix:
  command:
    default:
      execution:
        isolation:
          strategy: THREAD  # 隔离策略(THREADSEMAPHORE)
          thread:
            timeoutInMilliseconds: 2000  # 超时时间(默认1秒)
      circuitBreaker:
        enabled: true                   # 是否启用熔断
        requestVolumeThreshold: 5       # 触发熔断的最小请求数
        errorThresholdPercentage: 50    # 失败率阈值(%)
        sleepWindowInMilliseconds: 10000 # 熔断持续时间(ms)
  threadpool:
    default:
      coreSize: 10       # 线程池核心线程数
      maxQueueSize: 100  # 最大队列容量(-1表示禁用队列)
      queueSizeRejectionThreshold: 20  # 队列拒绝阈值

使用示例

基础熔断与降级​

@Service
public class UserService {

    @HystrixCommand(
        fallbackMethod = "getUserFallback",
        commandProperties = {
            // 自定义熔断规则(覆盖全局配置)
            @HystrixProperty(name = "circuitBreaker.requestVolumeThreshold", value = "3"),
            @HystrixProperty(name = "circuitBreaker.sleepWindowInMilliseconds", value = "5000")
        }
    )
    public String getUser(String userId) {
        if ("error".equals(userId)) {
            throw new RuntimeException("模拟服务失败");
        }
        return "User:" + userId;
    }

    // 降级方法(参数必须与原方法一致)
    public String getUserFallback(String userId) {
        return "Fallback User:" + userId;
    }
}

异步调用​

@HystrixCommand(fallbackMethod = "asyncFallback")
public Future<String> asyncGetUser(String userId) {
    return new AsyncResult<String>() {
        @Override
        public String invoke() {
            return "Async User:" + userId;
        }
    };
}

public String asyncFallback(String userId) {
    return "Async Fallback:" + userId;
}

忽略特定异常​

@HystrixCommand(
    fallbackMethod = "validateFallback",
    ignoreExceptions = { IllegalArgumentException.class }  # 此异常不触发熔断
)
public String validate(String input) {
    if (input.isEmpty()) {
        throw new IllegalArgumentException("输入为空");
    }
    return "Valid:" + input;
}

Hystrix+Feign Client 实现熔断

  1. 启用 Hystrix 和 Feign​
@SpringBootApplication
@EnableFeignClients    // 启用 Feign Client
@EnableHystrix          // 启用 Hystrix
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}
  1. 定义 Feign 接口及降级类​
// Feign Client 接口
@FeignClient(
    name = "user-service",
    url = "http://localhost:8081",
    fallback = UserServiceFallback.class  // 指定降级类
)
public interface UserServiceClient {
    @GetMapping("/users/{userId}")
    User getUser(@PathVariable String userId);
}

// 降级类实现 Feign 接口
@Component
public class UserServiceFallback implements UserServiceClient {
    @Override
    public User getUser(String userId) {
        return new User(userId, "Fallback User");
    }
}
  1. 配置熔断参数 (application.yml)​
feign:
  hystrix:
    enabled: true  # 启用 Hystrix 支持(默认 true,可省略)

hystrix:
  command:
    default:
      circuitBreaker:
        requestVolumeThreshold: 5       # 触发熔断的最小请求数
        errorThresholdPercentage: 50    # 失败率阈值(%)
        sleepWindowInMilliseconds: 10000 # 熔断持续时间(ms)
      execution:
        isolation:
          thread:
            timeoutInMilliseconds: 3000  # 超时时间
  1. 禁用 Hystrix 对特定 Feign Client​
@FeignClient(
    name = "order-service",
    url = "http://localhost:8082",
    configuration = FeignDisableHystrixConfig.class
)
public interface OrderServiceClient { ... }

// 自定义配置类
public class FeignDisableHystrixConfig {
    @Bean
    public Feign.Builder feignBuilder() {
        return Feign.builder();  // 不使用 Hystrix
    }
}

注意

  1. Hystrix 熔断器在 ​​打开状态(Open)​​ 后,默认经过 ​​5秒​​ 进入 ​​半开状态(Half-Open)
    可以通过 circuitBreaker.sleepWindowInMilliseconds 参数调整等待时间。
@HystrixCommand(
    commandProperties = {
        @HystrixProperty(
            name = "circuitBreaker.sleepWindowInMilliseconds", 
            value = "10000" // 10秒后进入半开状态
        )
    }
)
public String callService() { ... }
  1. Hystrix 的半开状态逻辑是固定不可配置​​的(Resilience4j可以进行配置),其行为如下:
    ​​探测请求数量​​:仅允许 ​​1次​​ 请求通过。
    ​​熔断关闭条件​​:
    如果探测请求 ​​成功​​,熔断器关闭,恢复服务。
    如果探测请求 ​​失败​​,熔断器重新打开,继续等待下一个 sleepWindowInMilliseconds 周期。

Resilience4j使用

依赖

<!-- Resilience4j 核心模块 -->
<dependency>
    <groupId>io.github.resilience4j</groupId>
    <artifactId>resilience4j-spring-boot2</artifactId>
    <version>1.7.1</version>
</dependency>
<!-- 支持注解方式(需 Spring AOP-->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

使用示例

熔断器

  1. 配置
resilience4j:
  circuitbreaker:
    instances:
      userService:
        failureRateThreshold: 50      # 失败率阈值(%)
        minimumNumberOfCalls: 5      # 触发统计的最小请求数
        slidingWindowType: COUNT_BASED  # 滑动窗口类型
        slidingWindowSize: 10         # 窗口大小(次数)
        waitDurationInOpenState: 10s  # 熔断持续时间
        permittedNumberOfCallsInHalfOpenState: 3  # 半开状态允许的请求数
        ignoreExceptions:             # 忽略特定异常(不触发熔断)
          - com.example.BusinessException
  1. 代码
@Service
public class UserService {

    @CircuitBreaker(name = "userService", fallbackMethod = "getUserFallback")
    public User getUser(String userId) {
        if (userId == null) {
            throw new RuntimeException("Invalid User ID");
        }
        return new User(userId, "Active");
    }

    // 降级方法(参数需一致,可添加异常参数)
    public User getUserFallback(String userId, Exception ex) {
        return new User(userId, "Fallback");
    }
}

Resilience4j+Feign Client 实现熔断

  1. 配置
resilience4j:
  circuitbreaker:
    instances:
      userService:  # 熔断器实例名(需与 Feign Client 名称对应)
        failureRateThreshold: 50     # 失败率阈值(%)
        minimumNumberOfCalls: 5    # 触发统计的最小请求数
        slidingWindowType: COUNT_BASED  # 滑动窗口类型
        slidingWindowSize: 10       # 窗口大小(次数)
        waitDurationInOpenState: 10s  # 熔断持续时间
        permittedNumberOfCallsInHalfOpenState: 3  # 半开状态允许的请求数
        recordExceptions:          # 触发熔断的异常类型
          - java.io.IOException
          - java.util.concurrent.TimeoutException
  1. 代码
@FeignClient(
    name = "userService",  // 与熔断器实例名一致
    url = "http://localhost:8081",
    fallback = UserServiceFallback.class // 直接指定降级类
)
public interface UserServiceClient {

    @GetMapping("/users/{userId}")
    User getUser(@PathVariable String userId);
}
  1. 熔断降级类
@Component
public class UserServiceFallback implements UserServiceClient {

    @Override
    public User getUser(String userId) {
        // 熔断降级逻辑
        return new User(userId, "Fallback User");
    }
}
  1. 在降级方法中捕获异常
@Component
public class UserServiceFallback implements UserServiceClient {

    @Override
    public User getUser(String userId, Throwable ex) {  // 添加 Throwable 参数
        log.error("Fallback triggered by: {}", ex.getMessage());
        return new User(userId, "Fallback User");
    }
}

重试(Retry)

  1. 配置
resilience4j:
  retry:
    instances:
      paymentService:
        maxAttempts: 3              # 最大重试次数
        waitDuration: 500ms         # 重试间隔
        retryExceptions:            # 触发重试的异常类型
          - java.io.IOException
  1. 代码
@Service
public class PaymentService {

    @Retry(name = "paymentService", fallbackMethod = "processPaymentFallback")
    public String processPayment(String orderId) {
        if (Math.random() > 0.3) {
            throw new IOException("Payment Gateway Error");
        }
        return "Payment Success: " + orderId;
    }

    public String processPaymentFallback(String orderId, Exception ex) {
        return "Fallback: " + orderId;
    }
}

限流(Rate Limiter)​

限制单位时间内的请求速率​​:通过 ​​令牌桶算法​​ 或 ​​漏桶算法​​,控制请求的流量。
​​实现方式​​:
​​令牌桶算法​​:以固定速率生成令牌,请求需获取令牌才能执行。
​​漏桶算法​​:强制请求以恒定速率执行,平滑处理突发流量。

  1. 配置
resilience4j:
  ratelimiter:
    instances:
      orderService:
        limitForPeriod: 5           # 时间窗口内允许的请求数
        limitRefreshPeriod: 1s      # 时间窗口长度
        timeoutDuration: 100ms      # 等待令牌的超时时间
  1. 代码
@Service
public class OrderService {

    @RateLimiter(name = "orderService", fallbackMethod = "createOrderFallback")
    public String createOrder(String productId) {
        return "Order Created: " + productId;
    }

    public String createOrderFallback(String productId, Exception ex) {
        return "Too Many Requests: " + productId;
    }
}

隔板(Bulkhead)​

​​限制同一时间的最大并发请求数​​:通过控制 ​​并发执行数量​​,防止资源耗尽

  1. 隔板的两种实现
    ​​信号量隔离(SemaphoreBulkhead)​​
    ​​原理​​:基于信号量(Semaphore)控制​​并发请求的最大数量​​。
    ​​行为​​:当并发请求数超过阈值时,立即拒绝新请求(抛出BulkheadFullException),无等待队列。
    ​​适用场景​​:轻量级、低延迟的并发控制,适用于I/O密集型或快速响应的操作。
    ​​线程池隔离(FixedThreadPoolBulkhead,已弃用)​​
    ​​原理​​:通过有界线程池和任务队列限制并发(类似Hystrix的线程隔离)。
    ​​行为​​:任务提交到线程池执行,队列满时拒绝新任务。
    ​​状态​​:Resilience4j官方已弃用此实现,推荐使用信号量隔离或结合其他异步库(如CompletableFuture)管理线程资源。
  2. 配置
resilience4j:
  bulkhead:
    instances:
      reportService:
        maxConcurrentCalls: 10      # 最大并发调用数
        maxWaitDuration: 10ms      # 等待进入隔板的超时时间

maxWaitDuration:

当隔板的并发请求数已满(达到 maxConcurrentCalls 限制)时,​​新请求可以等待其他请求释放资源的最大时间​​。
​​默认值​​:0(即不等待,直接拒绝请求,抛出 BulkheadFullException)。
​​单位​​:毫秒(例如 Duration.ofSeconds(2) 表示最多等待 2 秒)。
  1. 代码
@Service
public class ReportService {

    @Bulkhead(name = "reportService", fallbackMethod = "generateReportFallback")
    public String generateReport(String reportId) {
        return "Report: " + reportId;
    }

    public String generateReportFallback(String reportId, Exception ex) {
        return "System Busy: " + reportId;
    }
}

组合使用多个容错机制​

@Service
public class ProductService {
    // 同时应用熔断 + 重试 + 限流
    @CircuitBreaker(name = "productService", fallbackMethod = "fallback")
    @Retry(name = "productService", fallbackMethod = "fallback")
    @RateLimiter(name = "productService", fallbackMethod = "fallback")
    public String getProduct(String productId) {
        if (Math.random() > 0.5) {
            throw new RuntimeException("Service Error");
        }
        return "Product: " + productId;
    }

    public String fallback(String productId, Exception ex) {
        return "Fallback: " + productId;
    }
}

Hystrix vs Resilience4j

  1. 核心特性对比​
    在这里插入图片描述
  2. 功能深度对比​
    熔断器
    在这里插入图片描述
    容错模式​
    在这里插入图片描述
  3. 性能对比​
    资源消耗​​
    ​​Hystrix​​:线程池隔离需要为每个服务分配独立线程池,线程上下文切换开销大,适合资源充足的中型系统。
    ​​Resilience4j​​:信号量隔离无需创建线程,内存占用低,适合高并发场景和云原生环境。
    吞吐量测试​​
    ​​场景​​:1000并发请求,熔断触发条件相同。
    ​​结果​​:
    Resilience4j 吞吐量比 Hystrix 高 ​​30%~50%​​。
    Resilience4j 的 GC 停顿时间更短。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2384577.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C++23 容器从其他兼容范围的可构造性与可赋值性 (P1206R7)

文章目录 背景与动机提案内容与实现细节提案 P1206R7实现细节编译器支持 对开发者的影响提高灵活性简化代码向后兼容性 总结 C23标准引入了对容器构造和赋值的新特性&#xff0c;这些特性使得容器能够更灵活地从其他兼容范围初始化&#xff0c;并支持从范围赋值。这些改进由提案…

多通道振弦式数据采集仪MCU安装指南

设备介绍 数据采集仪 MCU集传统数据采集器与5G/4G,LoRa/RS485两种通信功能与一体的智能数据采集仪。该产品提供振弦、RS-485等的物理接口&#xff0c;能自动采集并存储多种自然资源、建筑、桥梁、城市管廊、大坝、隧道、水利、气象传感器的实时数据&#xff0c;利用现场采集的数…

SOC-ESP32S3部分:9-GPIO输入按键状态读取

飞书文档https://x509p6c8to.feishu.cn/wiki/L6IGwHKV6ikQ08kqwAwcAvhznBc 前面我们学习了GPIO的输出&#xff0c;GPIO输入部分其实也是一样的&#xff0c;这里我们使用按键作为GPIO输入例程讲解&#xff0c;分三步走。 查看板卡原理图&#xff0c;确定使用的是哪个GPIO查看G…

Ubuntu20.04的安装(VMware)

1.Ubuntu20.04.iso文件下载 下载网址&#xff1a;ubuntu-releases-20.04安装包下载_开源镜像站-阿里云 2.创建虚拟环境 2.1打开VMware与创建新虚拟机 点击创建新虚拟机 如果没下好可以点击稍后安装操作系统 选择linux版本选择Ubuntu 64位然后点击下一步。 注意这里需要选择一…

【论文阅读】LLaVA-OneVision: Easy Visual Task Transfer

LLaVA-OneVision: Easy Visual Task Transfer 原文摘要 研究背景与目标 开发动机&#xff1a; 基于LLaVA-NeXT博客系列对数据、模型和视觉表征的探索&#xff0c;团队整合经验开发了开源大型多模态模型 LLaVA-OneVision。 核心目标&#xff1a; 突破现有开源LMM的局限&#xf…

Spring Boot 项目多数据源配置【dynamic datasource】

前言&#xff1a; 随着互联网的发展&#xff0c;数据库的读写分离、数据迁移、多系统数据访问等多数据源的需求越来越多&#xff0c;我们在日常项目开发中&#xff0c;也不可避免的为了解决这个问题&#xff0c;本篇来分享一下在 Spring Boot 项目中使用多数据源访问不通的数据…

JAVA查漏补缺(2)

AJAX 什么是Ajax Ajax&#xff08;Asynchronous Javascript And XML&#xff09;&#xff0c;即是异步的JavaScript和XML&#xff0c;Ajax其实就是浏览器与服务器之间的一种异步通信方式 异步的JavaScript 它可以异步地向服务器发送请求&#xff0c;在等待响应的过程中&…

【Web前端】JavaScript入门与基础(二)

Javascript对象 什么是对象&#xff1f;对象&#xff08;object&#xff09;是 JavaScript 语言的核心概念&#xff0c;也是最重要的数据类型。简单说&#xff0c;对象就是一组“键值对”&#xff08;key-value&#xff09;的集合&#xff0c;是一种无序的复合数据集合。 var…

Electron+vite+vue3 从0到1搭建项目,开发Win、Mac客户端

随着前端技术的发展&#xff0c;出现了所谓的大前端。 大前端则是指基于前端技术延伸出来的各种终端平台及应用场景&#xff0c;包括APP、桌面端、手表终端、服务端等。 本篇文章主要是和大家一起学习一下使用Electron 如何打包出 Windows 和 Mac 所使用的客户端APP&#xff…

python打卡day34@浙大疏锦行

知识点回归&#xff1a; CPU性能的查看&#xff1a;看架构代际、核心数、线程数GPU性能的查看&#xff1a;看显存、看级别、看架构代际GPU训练的方法&#xff1a;数据和模型移动到GPU device上类的call方法&#xff1a;为什么定义前向传播时可以直接写作self.fc1(x) ①CPU性能查…

SOC-ESP32S3部分:8-GPIO输出LED控制

飞书文档https://x509p6c8to.feishu.cn/wiki/OSQWwh95niobqUkKyDQcVgsbnFg 这节课&#xff0c;我们将会以ESP32S3外设GPIO的使用为例&#xff0c;带大家学习如何从零开始学会ESP32外设的使用。 例如&#xff0c;这节课我们的需求是&#xff0c;需要通过GPIO控制指示灯的亮灭&…

05算法学习_59. 螺旋矩阵 II

05算法学习_59. 螺旋矩阵 II 05算法学习_59. 螺旋矩阵 II题目描述&#xff1a;个人代码&#xff1a;学习思路&#xff1a;第一种写法&#xff1a;题解关键点&#xff1a; 个人学习时疑惑点解答&#xff1a; 05算法学习_59. 螺旋矩阵 II 力扣题目链接: 59. 螺旋矩阵 II 题目描…

Linux `>`/`>>` 重定向操作符深度解析与高阶应用指南

Linux `>`/`>>` 重定向操作符深度解析与高阶应用指南 一、核心功能解析1. 基础重定向2. 标准流描述符二、高阶重定向技巧1. 多流重定向2. 文件描述符操作3. 特殊设备操作三、企业级应用场景1. 日志管理系统2. 数据管道处理3. 自动化运维四、安全与权限管理1. 防误操作…

【自定义类型-联合和枚举】--联合体类型,联合体大小的计算,枚举类型,枚举类型的使用

目录 一.联合体类型 1.1--联合体类型的声明 1.2--联合体的特点 1.3--相同成员的结构体和联合体对比 1.4--联合体大小的计算 1.5--联合体练习 二.枚举类型 2.1--枚举类型的声明 2.2--枚举类型的优点 2.3--枚举类型的使用 &#x1f525;个人主页&#xff1a;草莓熊Lotso…

李宏毅《深度学习》:Self-attention 自注意力机制

一&#xff0c;问题分析&#xff1a; 什么情况下需要使用self-attention架构&#xff0c;或者说什么问题是CNN等经典网络架构解决不了的问题&#xff0c;我们需要开发新的网络架构&#xff1f; 要解决什么问题《——》对应开发self-attention架构的目的&#xff1f; 1&#…

C++初阶-list的使用1

目录 1.std::list简介 2.成员函数 2.1构造函数的使用 2.2list::operator的使用 3.迭代器 4.容量 4.1list::empty函数的使用 4.2list::size函数的使用 4.3list::max_size函数的使用 5.元素访问 6.修饰符 6.1list::assign函数的使用 6.2push_back和pop_back和push_fr…

Python web 开发 Flask HTTP 服务

Flask 是一个轻量级的 Web 应用框架&#xff0c;它基于 Python 编写&#xff0c;特别适合构建简单的 Web 应用和 RESTful API。Flask 的设计理念是提供尽可能少的约定和配置&#xff0c;从而让开发者能够灵活地构建自己的 Web 应用。 https://andi.cn/page/622189.html

分享|16个含源码和数据集的计算机视觉实战项目

本文将分享16个含源码和数据集的计算机视觉实战项目。具体包括&#xff1a; 1. 人数统计工具 2. 颜色检测 3. 视频中的对象跟踪 4. 行人检测 5. 手势识别 6. 人类情感识别 7. 车道线检测 8. 名片扫描仪 9. 车牌识别 10. 手写数字识别 11.鸢尾花分类 12. 家庭照片人脸检测 13. 乐…

二十三、面向对象底层逻辑-BeanDefinitionParser接口设计哲学

一、引言&#xff1a;Spring XML配置的可扩展性基石 在Spring框架的演进历程中&#xff0c;XML配置曾长期作为定义Bean的核心方式。虽然现代Spring应用更倾向于使用注解和Java Config&#xff0c;但在集成第三方组件、兼容遗留系统或实现复杂配置逻辑的场景下&#xff0c;XML配…

[Vue]路由基础使用和路径传参

实际项目中不可能就一个页面&#xff0c;会有很多个页面。在Vue里面&#xff0c;页面与页面之间的跳转和传参会使用我们的路由: vue-router 基础使用 要使用我们需要先给我们的项目添加依赖:vue-router。使用命令下载: npm install vue-router 使用路由会涉及到下面几个对象:…