SpringCloud中Sentinel基础场景和异常处理

        Sentinel 是一个由 阿里巴巴 开源的分布式系统流量控制组件,专注于为微服务架构提供流量控制、熔断降级、系统负载保护等功能。它特别适用于高并发、高可用性的分布式系统,能够帮助开发者保护系统免于因流量过载、系统崩溃、依赖不可用等情况而导致的服务不可用。

一,Sentinel 的核心功能

  • 流量控制(Traffic Control)

    • 限流:在服务接口或者资源上进行流量限制,确保系统在高负载情况下不会崩溃。
    • QPS(每秒查询数):对每秒请求数进行限制,超过设定阈值则拒绝请求。
    • 并发线程数控制:控制在某个资源上同时处理请求的线程数,避免系统资源被过度消耗。
  • 熔断降级(Circuit Breaker)

    • 当依赖服务出现故障时,Sentinel 可以自动进行熔断处理,防止故障扩展,降低系统负担。
    • 根据定义的规则进行降级,比如当请求的失败率超过设定阈值时,自动进行服务降级处理。
    • 降级模式:可以配置为流量的百分比降级或者固定的线程数降级。
  • 系统负载保护(System Load Protection)

    • Sentinel 可以通过系统负载(如 CPU、内存、RT 等)来控制流量,确保在资源紧张时不会对系统产生更大的压力。
    • 可以根据系统的实时负载状况动态调整请求的流量,保证系统的稳定性。
  • 热点参数限流(Hot Spot Parameter Flow Control)

    • 限流不仅仅可以基于资源来进行,也可以基于请求中的参数来进行。例如,可以限制某些特定参数的请求数量。
  • 监控与报警

    • Sentinel 提供了实时监控能力,可以通过 Web 控制台或者日志进行流量、系统负载、异常等信息的监控。
    • 支持与外部监控系统(如 Prometheus、Grafana 等)集成,帮助开发者实时了解系统状态。
  • 集群模式

    • Sentinel 支持集群模式,适用于微服务架构中的多服务间的流量控制。
    • 支持将流量控制策略分布到整个集群中,保证整个系统的流量调度和熔断降级的一致性。

二,使用 Sentinel

添加依赖: 在 Spring Boot 项目中集成 Sentinel,首先需要在 pom.xml 中添加相关的依赖:
<dependency><groupId>com.alibaba.csp</groupId><artifactId>sentinel-core</artifactId><version>1.8.3</version>
</dependency>
使用 @SentinelResource 注解: 在方法上使用 @SentinelResource 注解来定义流量控制和熔断降级逻辑:
@SentinelResource(value = "createOrder")
@Override
public Order createOrder(Long productId, Long userId) {Product product = getProductFromRemoteWithLoadBalanceAnnotation(productId);// 使用 Feign 完成远程调用Product product = productFeignClient.getProductById(productId);Order order = new Order();order.setId(1L);// 总金额order.setTotalAmount(product.getPrice().multiply(new BigDecimal(product.getNum())));order.setUserId(userId);order.setNickName("zhangsan");return order;
}
  • @SentinelResource(value = "createOrder"):为 createOrder 方法添加了 Sentinel 的流量控制和熔断降级功能。这意味着该方法在执行时会受 Sentinel 管控,触发流量控制或熔断时,会执行相应的降级逻辑(如返回默认值或调用备用方法)。
  • productFeignClient.getProductById(productId):使用 Feign 远程调用其他服务获取产品信息。productFeignClient 是通过 Feign 客户端定义的接口,能够调用远程服务 getProductById 方法来获取指定 productId 的产品信息。

三,异常处理

其中异常处理流程主要分为三个大类:Web 接口的异常处理,@SentinelResource和 OpenFeign 调用的异常处理。涉及了不同的组件和处理方法:
 

1. Web 接口中的异常处理

在 Web 接口的异常处理中,主要使用 SentinelWebInterceptor 来拦截和处理请求中的异常。

1.1 SentinelWebInterceptor
  • 这是 Sentinel 中用于 Web 接口流量控制的拦截器。当请求的流量超出了预设的阈值(例如 QPS 限制),或者系统出现异常时,Sentinel 会自动触发 BlockException
  • 该拦截器可以捕获限流(流量控制)和降级的异常,并通过 BlockExceptionHandler 进行处理。
1.2 默认 BlockExceptionHandler
  • BlockExceptionHandler 是 Sentinel 默认的异常处理器。当发生限流或降级时,会调用默认的 BlockExceptionHandler 进行处理,通常它会返回一个简单的错误响应,提示用户请求被限流或服务降级。
  • 处理逻辑通常是返回一个自定义的错误消息,或者直接返回一个 500 或 429 状态码,表明请求被限制。
1.3 自定义 BlockExceptionHandler
  • 自定义异常处理器:如果默认的处理方式不符合需求,可以通过实现 BlockExceptionHandler 接口来创建自定义的异常处理逻辑。
  • 比如,你可以在自定义异常处理中加入详细的日志记录、告警机制、或者更复杂的错误响应格式。
import com.fasterxml.jackson.databind.ObjectMapper;
import com.alibaba.csp.sentinel.slots.block.BlockException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.PrintWriter;@Component
public class MyBlockExceptionHandler implements BlockExceptionHandler {private ObjectMapper objectMapper = new ObjectMapper();  // 用于将错误信息转换成 JSON 格式@Overridepublic void handle(HttpServletRequest request, HttpServletResponse response,String resourceName, BlockException e) throws Exception {// 获取响应的 PrintWriter 用于向客户端输出响应内容PrintWriter writer = response.getWriter();// 设置响应的内容类型为 JSONresponse.setContentType("application/json;charset=utf-8");// 创建一个自定义的错误信息对象 R,包含错误的代码和消息R error = R.error(500, resourceName + " 被Sentinel 限制了, 原因: " + e.getMessage());// 使用 ObjectMapper 将错误对象转换为 JSON 字符串String json = objectMapper.writeValueAsString(error);// 将 JSON 写入响应输出流writer.write(json);}
}
  • @Component 注解

    • 该类使用 @Component 注解标记为 Spring Bean,确保 Spring 能扫描到该类并进行自动注册。这个类实现了 BlockExceptionHandler 接口,作为 Sentinel 的自定义异常处理器。
  • ObjectMapper 实例化

    • ObjectMapper 是 Jackson 库中的一个类,用于将 Java 对象转换为 JSON 字符串。我们在 handle 方法中使用它来将错误信息对象 R 转换为 JSON 格式的字符串。
  • handle 方法

    • handle 方法会在 BlockException 发生时调用,这是 Sentinel 的异常处理方法。它接收以下参数:
      • request:当前请求对象。
      • response:当前响应对象,用于输出响应。
      • resourceName:触发限制的资源名称(如 API 名称)。
      • eBlockException,是 Sentinel 抛出的异常,表示请求被限流或熔断等原因阻塞。

 

2. @SentinelResource 注解中的异常(常用于控制器以外的类上)

在使用 @SentinelResource 注解时,我们可以为特定资源配置 流量控制降级熔断 等处理。它可以和 blockHandlerfallback 配合使用,确保服务在流量限制或异常发生时进行降级处理。

2.1 SentinelResourceAspect
  • SentinelResourceAspect 是 Sentinel 的一个切面,它会在方法执行时处理 流量控制(限流)和 异常降级。当触发 BlockException 时,会自动调用配置好的 blockHandler 方法来处理限流异常;如果发生其他异常(如业务逻辑失败),则会调用 fallback 方法进行降级处理。
2.2 blockHandler
  • blockHandler@SentinelResource 注解中的一个参数,用于指定当流量控制(限流)发生时,如何处理该异常。我们通过定义一个处理被阻断逻辑的blockHandler方法并使用注解将其与原始请求方法关联。resource标注的资源没有违反规则,则调用真实的业务逻辑去返回真实的数据,如果违反规则被渗透的限制了,则调用block handler指定的方法返回兜底数据,也就是control最终要么得到一个真实数据,要么得到一个兜底数据。
@SentinelResource(value = "createOrder", blockHandler = "createOrderFallback")
@Override
public Order createOrder(Long productId, Long userId) {// 使用 Feign 完成远程调用Product product = productFeignClient.getProductById(productId);Order order = new Order();order.setId(1L);// 总金额order.setTotalAmount(product.getPrice().multiply(new BigDecimal(product.getNum())));order.setUserId(userId);order.setNickName("zhangsan");order.setAddress("西安工业大学");// 远程查询商品列表order.setProductList(Arrays.asList(product));return order;
}// 降级回调
public Order createOrderFallback(Long productId, Long userId, BlockException e) {Order order = new Order();order.setId(0L);order.setTotalAmount(new BigDecimal("0"));order.setUserId(userId);order.setNickName("未注册用户");order.setAddress("异常信息: " + e.getClass());return order;
}
         这段代码保证了在请求被限流或发生异常时,能够回退到一个默认的响应,从而避免系统崩溃。具体通过 @SentinelResource 注解、blockHandler 机制实现了流量控制与异常降级处理。在控制后返回一个兜底回调数据。
2.3 fallback
  • fallback@SentinelResource 中的另一个参数,用于指定当服务出现异常时(如服务不可用或业务异常),如何进行服务降级处理。降级后的逻辑通常是提供备用数据或简单的错误响应。

     最终总结起来,我们的最佳实战用法就是SentinelResource一般标注在非control的这些层,你想要给哪些方法进行保护,你就加上这个注解,一旦违反规则以后,如果业务规定有兜底回调的数据,那么就使用block handler去来指定兜底回调,如果业务没规定有兜底回调,那我们也可以不用任何一种回调机制,直接让异常抛给全局,由项目的spring boot全局异常处理器进行处理。

3. OpenFeign 调用中的异常处理

OpenFeign 的调用中,Sentinel 同样提供了流量控制和熔断降级的功能。

3.1 SentinelFeign.builder()
  • SentinelFeign.builder() 用于构建 Feign 客户端时集成 Sentinel 的流量控制和熔断功能。当进行 Feign 调用时,Sentinel 会自动为调用方法设置流量控制规则。
  • 你可以通过 SentinelFeign.builder() 配置流量控制和降级策略,并为 Feign 方法定义 blockHandlerfallback
3.2 fallback
  • 在 Feign 调用中,fallback 用于指定当远程服务不可用或者请求超时时,如何降级。可以为每个 Feign 调用定义 fallback 方法,确保即使远程服务不可用,系统也能提供备用的返回值。

Feign 客户端接口

@FeignClient(value = "service-product", fallback = ProductFeignClientFallback.class) // feign客户端
public interface ProductFeignClient {@GetMapping("/product/{id}")Product getProductById(@PathVariable("id") Long id);
}

@FeignClient:定义了一个 Feign 客户端接口,value = "service-product" 表示它会向 service-product 服务发起请求,fallback 指定了降级,ProductFeignClientFallback.class,当远程调用失败时会触发该降级方法。 

Feign 客户端的降级处理:

@Component
public class ProductFeignClientFallback implements ProductFeignClient {@Overridepublic Product getProductById(Long id) {System.out.println("降级回调....");Product product = new Product();product.setId(id);product.setPrice(new BigDecimal("0"));product.setProductName("未知商品");product.setNum(0);return product;}
}

4. Spring Boot 异常处理

4.1 SpringBoot 异常处理
  • Sentinel 在 Spring Boot 中的集成也允许我们通过 全局异常处理 来捕获和处理来自 Sentinel 的 BlockException
  • 在 Spring Boot 中,使用 @ControllerAdvice 可以集中处理所有的异常,包括 BlockException。如果请求被限流或服务降级,Spring Boot 会捕获并处理这些异常,保证系统的高可用性

四,fallbackblockHandler 的区别

1. fallback
  • 目的fallback 用于处理 业务异常,例如远程服务不可用、请求超时等。它提供一种备用的处理方式,以保证系统的可用性。
  • 触发时机:当方法在正常执行过程中发生异常时(例如 getProductById 方法无法获取远程数据时),fallback 会被调用。
  • 使用场景:当服务请求失败时,fallback 会返回一个默认值或一个特定的错误响应。例如,产品查询失败时,返回一个“未知商品”的占位符。
2. blockHandler
  • 目的blockHandler 用于处理 流量控制相关的异常,例如当系统流量超出预定阈值时,Sentinel 会触发限流或熔断,调用 blockHandler 来处理这些流量控制异常。
  • 触发时机:当 流量控制(如限流、降级、熔断)触发时,blockHandler 会被调用,用来处理流量被阻塞的情况。
  • 使用场景:当请求被流量控制机制(如限流)阻塞时,blockHandler 会被触发,返回限流信息或适当的降级响应。例如,系统在高并发场景下超过阈值时,blockHandler 会返回“请求过多”的提示。

总结

  • fallback 主要是用于业务异常或服务不可用时提供备用逻辑,确保服务在出现故障时仍能返回合适的结果。

  • blockHandler 主要是处理流量控制相关的异常(如限流、熔断等),确保系统在流量过大或异常情况下不会崩溃。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

探索C语言中判断字符串循环移位关系的实现

在C语言的字符串处理中&#xff0c;判断两个字符串是否为循环移位关系是一个有趣且实用的问题。今天&#xff0c;我们就通过一段具体的代码来深入探讨这个问题的解决方案。 代码实现 代码逐行解析 预处理指令和头文件包含 #define _CRT_SECURE_NO_WARNINGS 用于禁用一些与安全…

Uniapp 原生组件层级过高问题及解决方案

文章目录 一、引言&#x1f3c5;二、问题描述&#x1f4cc;三、问题原因❓四、解决方案&#x1f4af;4.1 使用 cover-view 和 cover-image4.2 使用 subNVue 子窗体4.3 动态隐藏原生组件4.4 使用 v-if 或 v-show 控制组件显示4.5 使用 position: fixed 布局 五、总结&#x1f38…

【Jenkins流水线搭建】

Jenkins流水线搭建 01、SpringBoot项目 - Jenkins基于Jar持续集成搭建文档基于手动方式发布项目基于dockerfile基于jenkins + dockerfile + jenkinsfile +pieline基于jenkins + jar方式的发布01、环境说明01、准备项目02、准备服务器03、安装git04、安装jdk1.805、安装maven依赖…

python包的管理

管理python包 python能跻身最欢迎编程语言前列的一个主要原因是python有着活跃的社区提供丰富的包&#xff0c;诸如numpy&#xff0c;pandas&#xff0c;scikit-learn等等。 python的包都存放PyPI中&#xff0c;PyPI即Python Package Index&#xff0c;是python的软件仓库。所…

2025常用的SEO工具有哪些?

在互联网时代&#xff0c;如何让自己的网站或内容脱颖而出&#xff0c;成为许多企业和个人站长们最关注的问题。而在这个过程中&#xff0c;SEO&#xff08;搜索引擎优化&#xff09;作为一种有效的提升网站曝光度和吸引流量的手段&#xff0c;已经成为了网站运营的核心之一。对…

消息中间件深度剖析:以 RabbitMQ 和 Kafka 为核心

在现代分布式系统和微服务架构的构建中&#xff0c;消息中间件作为一个不可或缺的组件&#xff0c;承担着系统间解耦、异步处理、流量削峰、数据传输等重要职能。尤其是在面临大规模并发、高可用性和可扩展性需求时&#xff0c;如何选择合适的消息中间件成为了开发者和架构师们…

深入解析SVG图片原理:从基础到高级应用

文章目录 引言一、SVG基础概念1.1 什么是SVG&#xff1f;1.2 SVG的优势 二、SVG的基本结构2.1 SVG文档结构2.2 常用SVG元素 三、SVG的工作原理3.1 坐标系与变换3.2 路径与曲线3.3 渐变与滤镜 四、SVG的高级应用4.1 动画与交互4.2 数据可视化4.3 响应式设计 五、SVG的优化与性能…

【读点论文】Rewrite the Stars将svm的核技巧映射到高维空间,从数理逻辑中丰富特征维度维度

Rewrite the Stars Abstract 最近的研究已经引起了人们对网络设计中“星形运算”(逐元素乘法)的未开发潜力的关注。虽然直观的解释比比皆是&#xff0c;但其应用背后的基本原理在很大程度上仍未被探索。我们的研究试图揭示星形操作在不扩大网络的情况下将输入映射到高维非线性…

C++中常用的十大排序方法之4——希尔排序

成长路上不孤单&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a; 【&#x1f60a;///计算机爱好者&#x1f60a;///持续分享所学&#x1f60a;///如有需要欢迎收藏转发///&#x1f60a;】 今日分享关于C中常用的排序方法之4——希尔排序的相…

初阶c语言(练习题,猜随机数,关机程序)

目录 第一题&#xff0c;使用函数编写一个随机数&#xff0c;然后自己猜&#xff0c;猜随机数 第二道题&#xff08;关机程序&#xff09; 实现代码&#xff08;关机程序&#xff09; 实现代码&#xff08;猜数字&#xff09; 前言&#xff1a; 学习c语言&#xff0c;学习…

离线量化算法和工具 --学习记录1

离线量化算法和工具 一、离线量化的基础概念1.1、基本流程1.2、量化的优点和缺点1.3、如何生产一个硬件能跑的量化模型1.4、PTQ的概念以及和QAT的区别1.5、离线量化的标准流程1.6、校准数据的选择1.7、量化模式的选择1.8、校准方式的选择1.9、量化算法的选择1.10、写入量化参数…

封装一个sqlite3动态库

作者&#xff1a;小蜗牛向前冲 名言&#xff1a;我可以接受失败&#xff0c;但我不能接受放弃 如果觉的博主的文章还不错的话&#xff0c;还请点赞&#xff0c;收藏&#xff0c;关注&#x1f440;支持博主。如果发现有问题的地方欢迎❀大家在评论区指正 目录 一、项目案例 二…

ROS进阶:使用URDF和Xacro构建差速轮式机器人模型

前言 本篇文章介绍的是ROS高效进阶内容&#xff0c;使用URDF 语言&#xff08;xml格式&#xff09;做一个差速轮式机器人模型&#xff0c;并使用URDF的增强版xacro&#xff0c;对机器人模型文件进行二次优化。 差速轮式机器人&#xff1a;两轮差速底盘由两个动力轮位于底盘左…

移远通信边缘计算模组成功运行DeepSeek模型,以领先的工程能力加速端侧AI落地

近日&#xff0c;国产大模型DeepSeek凭借其“开源开放、高效推理、端侧友好”的核心优势&#xff0c;迅速风靡全球。移远通信基于边缘计算模组SG885G&#xff0c;已成功实现DeepSeek模型的稳定运行&#xff0c;并完成了针对性微调。 目前&#xff0c;该模型正在多款智能终端上进…

resultType,jdbcType,parameterType区别

1. resultType 用途&#xff1a; 用于定义 SQL 查询结果的返回类型。 直接将查询结果映射到指定的 Java 类型&#xff08;基本类型、POJO 或 Map&#xff09;。 特点&#xff1a; 要求数据库字段名与 Java 对象的属性名完全一致&#xff08;或通过别名匹配&#xff09;。 …

字符设备驱动开发

驱动就是获取外设、传感器数据和控制外设。数据会提交给应用程序。 Linux 驱动编译既要编写一个驱动&#xff0c;还要编写一个简单的测试应用程序。 而单片机下驱动和应用都是放在一个文件里&#xff0c;也就是杂在一块。而 Linux 则是分开了。 一、字符设备驱动开发流程 Lin…

【免费送书活动】《MySQL 9从入门到性能优化(视频教学版)》

本博主免费赠送读者3本书&#xff0c;书名为《MySQL 9从入门到性能优化&#xff08;视频教学版&#xff09;》。 《MySQL 9从入门到性能优化&#xff08;视频教学版&#xff09;&#xff08;数据库技术丛书&#xff09;》(王英英)【摘要 书评 试读】- 京东图书 这本书已经公开…

UE求职Demo开发日志#32 优化#1 交互逻辑实现接口、提取Bag和Warehouse的父类

1 定义并实现交互接口 接口定义&#xff1a; // Fill out your copyright notice in the Description page of Project Settings.#pragma once#include "CoreMinimal.h" #include "UObject/Interface.h" #include "MyInterActInterface.generated.h…

DeepSeek 指导手册(入门到精通)

第⼀章&#xff1a;准备篇&#xff08;三分钟上手&#xff09;1.1 三分钟创建你的 AI 伙伴1.2 认识你的 AI 控制台 第二章&#xff1a;基础对话篇&#xff08;像交朋友⼀样学交流&#xff09;2.1 有效提问的五个黄金法则2.2 新手必学魔法指令 第三章&#xff1a;效率飞跃篇&…

Next.js【详解】获取数据(访问接口)

Next.js 中分为 服务端组件 和 客户端组件&#xff0c;内置的获取数据各不相同 服务端组件 方式1 – 使用 fetch export default async function Page() {const data await fetch(https://api.vercel.app/blog)const posts await data.json()return (<ul>{posts.map((…