系列文章目录
1、父工程创建
2、支付模块构建和热部署
3、消费者订单模块
4、服务注册中心-Eureka
5、zookeeper没学习
6、服务注册中心-Consul
7、Eureka、Consul异同
8、服务调用-Ribbon
9、服务调用-OpenFeign
10、服务降级-Hystrix
11、服务降级-Hystrix(二)
12、服务熔断-Hystrix
13、服务网关-Gateway
14-17 在git上做配置中心,没有学习
17、请求链路跟踪-Sleuth
18、Spring Cloud Alibaba-Nacos注册中心与配置中心
19、Spring Cloud Alibaba-Nacos集群和持久化配置
20、Sentinel流控
21、Sentinel熔断降级、热点key限流
22、SentinelResource配置
23、Sentinel 服务熔断与持久化
1. 按资源名称限流+后续处理
1.1 前提
- nacos 启动
- Sentinel 启动
1.2 修改module
修改cloudalibaba-sentinel-service8401
###1.2.1 POM
<!-- commons -->
<dependency>
<groupId>com.chzu</groupId>
<artifactId>cloud-commons</artifactId>
<version>${project.version}</version>
</dependency>
1.2.2 YML
不需要更改
1.2.3 业务类
新增 RateLimitController
@RestController
public class RateLimitController {
@GetMapping("/byResource")
@SentinelResource(value = "byResource",blockHandler = "handleException")
public CommonResult byResource() {
return new CommonResult(200,"按资源名称限流测试OK",new Payment(2020L,"serial001"));
}
public CommonResult handleException(BlockException exception) {
return new CommonResult(444,exception.getClass().getCanonicalName()+"\t 服务不可用");
}
}
1.2.4 配置流控规则
表示1秒钟内查询次数大于1,就到我们自定义的限流方法
测试:
http://localhost:8401/byResource
- 一秒点击一下
- 一秒点击多下,返回如下
{"code":444,"message":"com.alibaba.csp.sentinel.slots.block.flow.FlowException\t 服务不可用","data":null}
2. 按URL地址限流+后续处理
通过访问的URL来限流,会返回Sentinel自带默认的限流处理信息
2.1修改module
2.1.1 业务类
RateLimitController添加一个方法
@RestController
public class RateLimitController
{
@GetMapping("/rateLimit/byUrl")
@SentinelResource(value = "byUrl")
public CommonResult byUrl()
{
return new CommonResult(200,"按url限流测试OK",new Payment(2020L,"serial002"));
}
}
2.1.2 配置流控规则
表示1秒钟内查询次数大于1,会限流。
测试:
快速点击http://localhost:8401/rateLimit/byUrl
返回: Blocked by Sentinel (flow limiting)
疑问:我在当时学习的时候
还想着因为Controller中的这个方法都没有blockHandler = ""指定返回,所以会返回Sentinel默认的处理信息
但是我在个这个方法添加blockHandler之后,他还是会返回Sentinel默认的处理信息。
2.1.3 存在问题
- 限流返回信息为系统默认,没有体现我们自己的业务要求。
- 我们自定义的处理方法又和业务代码耦合在一块,不直观。
- 每个业务方法都添加—个兜底的,那代码膨胀加剧。
- 全局统—的处理方法没有体现。
3. 用户自定义限流处理逻辑
3.1 修改module
3.1.1 业务类
创建CustomerBlockHandler类用于自定义限流处理逻辑
/**
* @author lwd
* @description 自定义限流逻辑
*/
public class CustomerBlockHandler {
public static CommonResult handlerException(BlockException exception) {
return new CommonResult(4444,"按客戶自定义,global handlerException----1");
}
public static CommonResult handlerException2(BlockException exception) {
return new CommonResult(4444,"按客戶自定义,global handlerException----2");
}
}
RateLimitController增加一个方法
@RestController
public class RateLimitController {
.....
@GetMapping("/rateLimit/customerBlockHandler")
@SentinelResource(value = "customerBlockHandler",
blockHandlerClass = CustomerBlockHandler.class,//自定义限流处理类
blockHandler = "handlerException2")//
public CommonResult customerBlockHandler()
{
return new CommonResult(200,"按客戶自定义",new Payment(2020L,"serial003"));
}
}
测试:
访问http://localhost:8401/rateLimit/customerBlockHandler。
连续点击多次,我们自定义兜底方法的字符串信息就返回到前端
3.1.2 注意点
CustomerBlockHandler
public static CommonResult handlerException(BlockException exception) {
return new CommonResult(4444,"按客戶自定义,global handlerException----1");
}
- CustomerBlockHandler中的方法需要是静态的,否则限流时返回ErrorPage
- CustomerBlockHandler中方法需要参数BlockException exception,否则限流时返回ErrorPage
- 不要使用URL的方式限流,那样不会调用自定义的逻辑。
4. 注解的更多属性说明
@SentinelResource 注解
注意:注解方式埋点不支持 private 方法。@SentinelResource 用于定义资源,并提供可选的异常处理和 fallback 配置项。 @SentinelResource 注解包含以下属性:
value:资源名称,必需项(不能为空)
entryType:entry 类型,可选项(默认为 EntryType.OUT)
blockHandler / blockHandlerClass: blockHandler 对应处理 BlockException的函数名称,可选项。blockHandler 函数访问范围需要是public,返回类型需要与原方法相匹配,参数类型需要和原方法相匹配并且最后加一个额外的参数,类型为BlockException。blockHandler 函数默认需要和原方法在同一个类中。若希望使用其他类的函数,则可以指定blockHandlerClass 为对应的类的 Class 对象,注意对应的函数必需为 static 函数,否则无法解析。
fallback / fallbackClass:fallback 函数名称,可选项,用于在抛出异常的时候提供 fallback处理逻辑。fallback 函数可以针对所有类型的异常(除了 exceptionsToIgnore 里面排除掉的异常类型)进行处理。fallback 函数签名和位置要求:
- 返回值类型必须与原函数返回值类型一致;
- 方法参数列表需要和原函数一致,或者可以额外多一个 Throwable 类型的参数用于接收对应的异常。
- fallback函数默认需要和原方法在同一个类中。若希望使用其他类的函数,则可以指定 fallbackClass 为对应的类的 Class对象,注意对应的函数必需为 static 函数,否则无法解析。
defaultFallback(since 1.6.0):默认的fallback 函数名称,可选项,通常用于通用的 fallback 逻辑(即可以用于很多服务或方法)。默认 fallback函数可以针对所有类型的异常(除了 exceptionsToIgnore 里面排除掉的异常类型)进行处理。若同时配置了 fallback 和defaultFallback,则只有 fallback 会生效。defaultFallback 函数签名要求:
- 返回值类型必须与原函数返回值类型一致;
- 方法参数列表需要为空,或者可以额外多一个 Throwable 类型的参数用于接收对应的异常。
- defaultFallback 函数默认需要和原方法在同一个类中。若希望使用其他类的函数,则可以指定 fallbackClass为对应的类的 Class 对象,注意对应的函数必需为 static 函数,否则无法解析。
exceptionsToIgnore(since1.6.0):用于指定哪些异常被排除掉,不会计入异常统计中,也不会进入 fallback 逻辑中,而是会原样抛出。1.8.0 版本开始,defaultFallback 支持在类级别进行配置。
注:1.6.0 之前的版本 fallback 函数只针对降级异常(DegradeException)进行处理,不能针对业务异常进行处理。
特别地,若 blockHandler 和 fallback 都进行了配置,则被限流降级而抛出 BlockException 时只会进入blockHandler 处理逻辑。若未配置 blockHandler、fallback 和defaultFallback,则被限流降级时会将 BlockException 直接抛出(若方法本身未定义 throws BlockException 则会被 JVM 包装一层 UndeclaredThrowableException)。link