【SpringCloud】学习笔记之微服务保护

【SpringCloud】学习笔记之微服务保护

1.雪崩问题及解决方案

1.1 雪崩问题概述

在微服务中,服务间的调用关系错综复杂,一个微服务往往依赖于多个其他微服务。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-w7fBIURJ-1671787346275)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\1533829099748.png)]

如图,如果服务提供者I发生了故障,那么当前的调用I的服务消费者因为依赖于服务I,因此也会被阻塞。此时,其他不依赖于服务I的业务似乎不会受到影响。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-2aQgZuEn-1671787346276)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\1533829198240.png)]

但是,依赖服务I的业务请求被阻塞,用户不会得到响应,则tomcat的这个线程不会释放,于是越来越多的用户请求到来,越来越多的线程会阻塞:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-SSfrJVQO-1671787346277)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\1533829307389.png)]

服务器支持的线程和并发数有限,请求一直阻塞,会导致服务器资源耗尽,从而导致该服务的其他业务都不可用,那么当前服务也就不可用了。

那么,依赖于当前服务的其他服务随着时间的推移,最终也都会变得不可用,形成级联失败,雪崩由此发生。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-lkIOij7P-1671787346277)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210715172710340.png)]


1.2 雪崩问题解决方案

解决雪崩问题有如下常见的四种方式:

  • 超时处理
  • 舱壁模式
  • 熔断降级
  • 流量控制

1.2.1 超时处理

超时处理:设定超时时间,请求一旦超过一定时间没有响应就返回错误信息,不会无休止的等待而空占线程资源。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-lCUN2puQ-1671787346278)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210715172820438.png)]


1.2.2 舱壁模式/线程隔离

舱壁模式设计思想来源于船舱的设计:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hrEsrWJZ-1671787346278)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210715172946352.png)]

船厂都会被多个隔板分离为多个独立空间,当船体破损时,只会导致部分空间进入,将故障控制在一定范围内,避免整个船体都被淹没。

与此类似,我们可以限定服务中每个业务所用的最大线程数,从而避免耗尽整个tomcat的资源,因此也叫线程隔离

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-oY9KM6pQ-1671787346279)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210715173215243.png)]


1.2.3 熔断降级/断路器

断路器模式:由断路器统计业务执行的异常比例,如果超出阈值则会熔断该业务,拦截访问该业务的一切请求。

断路器会统计访问某个服务的请求数量,异常比例:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-exE76RWX-1671787346285)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210715173327075.png)]

当发现访问服务D的请求异常比例超过一定限额,则认为服务D由导致雪崩的风险,会拦截访问服务D的一切请求,形成熔断

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-RVtBv88O-1671787346286)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210715173428073.png)]


1.2.4 流量控制

流量控制:限制业务访问的QPS,避免服务因流量的突增而故障。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-EiARtxym-1671787346286)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210715173555158.png)]


1.3 总结

什么是雪崩问题?

微服务之间相互调用,因为调用链中的一个服务故障,引起整个链路都无法访问的情况。

可以认为:

  • 限流是对服务的保护,避免因瞬间高并发流量而导致服务故障,进而避免雪崩。是一种预防措施。
  • 超时处理、线程隔离、熔断降级是在部分服务故障时,将故障控制在一定范围,避免雪崩。是一种补救措施。

1.4 Sentinel

1.4.1 Sentinel简介

Sentinel是阿里巴巴开源的一款微服务流量控制组件。官网地址:https://sentinelguard.io/zh-cn/index.html

Sentinel 具有以下特征:

丰富的应用场景:Sentinel 承接了阿里巴巴近 10 年的双十一大促流量的核心场景,例如秒杀(即突发流量控制在系统容量可以承受的范围)、消息削峰填谷、集群流量控制、实时熔断下游不可用应用等。

完备的实时监控:Sentinel 同时提供实时的监控功能。您可以在控制台中看到接入应用的单台机器秒级数据,甚至 500 台以下规模的集群的汇总运行情况。

广泛的开源生态:Sentinel 提供开箱即用的与其它开源框架/库的整合模块,例如与 Spring Cloud、Dubbo、gRPC 的整合。您只需要引入相应的依赖并进行简单的配置即可快速地接入 Sentinel。

完善的 SPI 扩展点:Sentinel 提供简单易用、完善的 SPI 扩展接口。您可以通过实现扩展接口来快速地定制逻辑。例如定制规则管理、适配动态数据源等。


1.4.2 Sentinel运行

将下载好的sentinel放到任意非中文目录中,执行以下命令:

java -jar sentinel-dashboard-1.8.1.jar

如果要修改Sentinel的默认端口、账户、密码,可以通过下列配置:

配置项默认值说明
server.port8080服务端口
sentinel.dashboard.auth.usernamesentinel默认用户名
sentinel.dashboard.auth.passwordsentinel默认密码

例如,修改端口:

java -Dserver.port=8090 -jar sentinel-dashboard-1.8.1.jar

3)访问

访问http://localhost:8080页面,就可以看到sentinel的控制台了:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tDziJmL2-1671787346287)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210715190827846.png)]

需要输入账号和密码,默认都是:sentinel

登录后,发现一片空白,后台还什么都没有。


1.5 微服务整合Sentinel

在order-service中整个sentinel,连接其控制台。

1.5.1 引入依赖

引入sentinel依赖:

<!--sentinel-->
<dependency>
    <groupId>com.alibaba.cloud</groupId> 
    <artifactId>spring-cloud-starter-alibaba-sentinel</artifactId>
</dependency>

1.5.2 配置控制台

在配置文件中添加如下内容:

server:
  port: 8088
spring:
  cloud: 
    sentinel:
      transport:
        dashboard: localhost:8080

1.5.3 测试

在浏览器中访问order-service的任意controller接口,这样才能触发sentinel的监控,sentinel控制台才会有内容。

然后访问sentinel控制台,如图:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-syJ2vBDN-1671787346287)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222194430626.png)]


2. 流量控制

雪崩问题虽然有四种方案,但是限流是避免服务因突发的流量而发生故障,是对微服务雪崩问题的预防。

2.1 簇点链路

当请求进入微服务时,首先会访问DispatcherServlet,然后进入Controller、Service、Mapper,这样的一个调用链就叫做簇点链路。簇点链路中被监控的每一个接口就是一个资源

默认情况下sentinel会监控SpringMVC的每一个端点(Endpoint,也就是controller中的方法),因此SpringMVC的每一个端点(Endpoint)就是调用链路中的一个资源。

**例如:**我们刚才访问的order-service中的OrderController中的端点:/order/{orderId}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-91qggEya-1671787346288)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222195605952.png)]

流控、熔断等都是针对簇点链路中的资源来设置的,因此我们可以点击对应资源后面的按钮来设置规则:

  • 流控:流量控制
  • 降级:熔断降级
  • 热点:热点参数限流,是限流的一种
  • 授权:请求的权限控制

2.2 快速入门

点击资源/order/{orderId}后面的流控按钮,就可以弹出表单。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-G9IN6YFA-1671787346289)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210715191757319.png)]

表单中可以填写限流规则,如下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-INkxNpm8-1671787346289)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210715192010657.png)]

其含义是限制 /order/{orderId}这个资源的单机QPS为1,即每秒只允许1次请求,超出的请求会被拦截并报错。


**需求:**给 /order/{orderId}这个资源设置流控规则,QPS不能超过 5,然后测试。

实践:

1)点击对应的”流控“按钮,将单机阈值修改为5,然后点击“新增”按钮。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-1Cs1BrK7-1671787346290)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222200041014.png)]

2)然后我们会看到如下内容:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-mK3QczCh-1671787346291)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222200135086.png)]

3)使用jmeter进行测试。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-oO86iHqB-1671787346291)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222201215245.png)]

20个用户发的1次请求在2s内完成,QPS为10,而我们设置的QPS为5.运行这个测试,结果如下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-BocKzMhr-1671787346291)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222201331013.png)]


2.3 流控模式

在添加先流规则时,点击高级选项,可以选择三种流控模式

  • 直接:统计当前资源的请求,触发阈值时对当前资源直接限流,也是默认的模式。
  • 关联:统计与当前资源相关的另一个资源,另一个资源触发阈值时,对当前资源限流。
  • 链路:统计从指定链路访问到本资源的请求,触发阈值时,对指定链路限流。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hwHHc8iF-1671787346292)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210715201827886.png)]

我们刚才的“快速入门”就是直接模式。


2.3.1 关联模式

关联模式:统计与当前资源相关的另一个资源,另一个资源触发阈值时,对当前资源限流。

配置规则

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Q30Ls9Kk-1671787346292)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210715202540786.png)]

语法说明:当/write资源访问量触发阈值时,就会对/read资源限流,避免影响/write资源。

使用场景:比如用户支付时需要修改订单状态,同时用户要查询订单。查询和修改操作会争抢数据库锁,产生竞争。业务需求是优先支付和更新订单的业务,因此当修改订单业务触发阈值时,需要对查询订单业务限流。

需求

  • 在OrderController新建两个端点:/order/query和/order/update,无需实现业务
  • 配置流控规则,当/order/ update资源被访问的QPS超过5时,对/order/query请求限流

实践

1)定义/order/query端点,模拟订单查询

@GetMapping("/query")
    public String queryOrder(){
        return "查询订单成功";
    }

2)定义/order/update端点,模拟订单更新

 @GetMapping("/update")
    public String updateOrder(){
        return "修改订单成功";
    }

3)重启服务,访问接口,然后查看sentinel控制台的簇点链路:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-LoECaTH9-1671787346293)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222211432150.png)]

4)配置流控规则

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-q0OmlAQm-1671787346293)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222211702341.png)]

5)Jmeter测试

对/order/update接口进行测试,参数如下:(QPS为10,大于设定值5)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-uEZZyeXS-1671787346294)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222211940208.png)]

测试结果如下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bdOBLqrA-1671787346294)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222212033690.png)]

所有请求都成功了,我们再去访问/order/query接口。如图:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fdDvdjQS-1671787346294)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222211916795.png)]

说明成功得对/order/query接口进行了限流。

6)总结

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-O4wxTTEQ-1671787346295)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210716103143002.png)]


2.3.2 链路模式

链路模式:只针对从指定链路访问到本资源的请求做统计,判断是否超过阈值。

配置示例

例如有两条请求链路:

  • /test1 --> /common

  • /test2 --> /common

如果只希望统计从/test2进入到/common的请求,则可以这样配置:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KC61mmto-1671787346295)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716103536346.png)]

实战案例

需求:有查询订单和创建订单业务,两者都需要查询商品。针对从查询订单进入到查询商品的请求统计,并设置限流。

分析:查询订单和常见订单时都要查询商品信息,但是查询商品这个业务肯定有自己的并发上限。如果查询订单的并发过大,则一定会影响创建订单的业务。所以我们需要对查询订单做一个限流,以确保创建订单的业务能够正常运行。

步骤

  1. 在OrderService中添加一个 queryGoods 方法,不用实现业务。
  2. 在OrderController中,改造/order/query端点,调用OrderService中的 queryGoods 方法。
  3. 在OrderController中添加一个/order/save的端点,调用OrderService的 queryGoods 方法。
  4. 给queryGoods设置限流规则,从/order/query进入queryGoods的方法限制必须小于2.

实现

1)添加查询商品的方法

在order-service服务中,给OrderService类添加一个queryGoods方法:

public void queryGoods(){
    System.err.println("查询商品");
}

2)查询订单时,查询商品

修改/order/query端点的业务逻辑。

@GetMapping("/query")
public String queryOrder(){
    //查询商品
    orderService.queryGoods();
    //查询订单
    System.out.println("查询订单");
    return "查询订单成功";
}

3)新增订单,查询商品

在OrderController中新增一个/query/save端点:

@GetMapping("/save")
public String saveOrder(){
    //查询商品
    orderService.queryGoods();
    //新增订单
    System.out.println("新增订单");
    return "新增订单成功";
}

4)给查询商品添加资源标记

默认情况下,OrderService中的方法是不被Sentinel监控的,需要我们自己通过注解来标记要监控的方法。

给OrderService的queryGoods方法添加@SentinelResource注解:

@SentinelResource("goods")
public void queryGoods(){
    System.err.println("查询商品");
}

链路模式中,是对不同来源的两个链路做监控。但是sentinel默认会给进入SpringMVC的所有请求设置同一个root资源,会导致链路模式失效。

我们需要关闭这种对SpringMVC的资源聚合,修改order-service服务的application.yml文件:

spring:
  cloud:
    sentinel:
      web-context-unify: false # 关闭context整合

重启服务,访问/order/query和/order/save,可以查看到sentinel的簇点链路规则中,出现了新的资源:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-sDy3kkur-1671787346296)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222215147529.png)]

5)添加流控规则

点击goods资源后面的流控按钮,在弹出的表单填写如下信息:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-egzY7oGf-1671787346296)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222215120909.png)]

只统计从/order/query进入/goods的资源,QPS阈值为2,超出则被限流。

6)Jmeter测试

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ifldGzBG-1671787346296)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222215503209.png)]

查看 /order/query 的结果树:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KiY84pZm-1671787346297)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222215545680.png)]

因为设置的阈值为2,而我们的qps为4,所以每秒4次的请求只能成功2次,剩余失败。

查看 /order/save 的结果树:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-n3WpXFhs-1671787346297)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222215735893.png)]

全部成功。


2.4 流控效果

在流控的高级选项中,还有一个流控效果选项:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-r6on4o8A-1671787346298)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716110225104.png)]

流控效果是指请求达到流控阈值时应该采取的措施,包括三种:

  • 快速失败:达到阈值后,新的请求会被立即拒绝并抛出FlowException异常。是默认的处理方式。
  • warm up:预热模式,对超出阈值的请求同样是拒绝并抛出异常。但这种模式阈值会动态变化,从一个较小值逐渐增加到最大阈值。
  • 排队等待:让所有的请求按照先后次序排队执行,两个请求的间隔不能小于指定时长

2.4.1 Warm Up

阈值一般是一个微服务能承担的最大QPS,但是一个服务刚刚启动时,一切资源尚未初始化(冷启动),如果直接将QPS跑到最大值,可能导致服务瞬间宕机。

warm up也叫预热模式,是应对服务冷启动的一种方案。请求阈值初始值是 maxThreshold / coldFactor,持续指定时长后,逐渐提高到maxThreshold值。而coldFactor的默认值是3.

例如,我设置QPS的maxThreshold为10,预热时间为5秒,那么初始阈值就是 10 / 3 ,也就是3,然后在5秒后逐渐增长到10.

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-E02ntFzO-1671787346298)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716110629796.png)]

案例

需求:给/order/{orderId}这个资源设置限流,最大QPS为10,利用warm up效果,预热时长为5秒

实现

1)配置流控规则:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KFyj0LqX-1671787346299)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222220337943.png)]

2)Jmeter测试

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-214Qq32w-1671787346299)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222221026311.png)]

查看结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-DZOsBNrk-1671787346300)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222220759511.png)]

刚刚启动时,只有三个请求能够成功,说明此时qps被限定为3,随着时间推移,成功的数量越来越多,说明qps越来越靠近我们设定的值,5s过后,gps为10.所有的请求都成功。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bK5w6EpN-1671787346300)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222221044028.png)]


2.4.2 排队等待

当请求超过QPS阈值时,快速失败和warm up 会拒绝新的请求并抛出异常。

而排队等待则是让所有请求进入一个队列中,然后按照阈值允许的时间间隔依次执行。后来的请求必须等待前面执行完成,如果请求预期的等待时间超出最大时长,则会被拒绝。

工作原理

例如:QPS = 5,意味着每200ms处理一个队列中的请求;timeout = 2000,意味着预期等待时长超过2000ms的请求会被拒绝并抛出异常。

那什么叫做预期等待时长呢?

比如现在一下子来了12 个请求,因为每200ms执行一个请求,那么:

  • 第6个请求的预期等待时长 = 200 * (6 - 1) = 1000ms
  • 第12个请求的预期等待时长 = 200 * (12-1) = 2200ms

现在,第1秒同时接收到10个请求,但第2秒只有1个请求,此时QPS的曲线这样的:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-OPNZaQZR-1671787346300)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716113147176.png)]

如果使用队列模式做流控,所有进入的请求都要排队,以固定的200ms的间隔执行,QPS会变的很平滑:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-SUrTfBzO-1671787346301)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716113426524.png)]

平滑的QPS曲线,对于服务器来说是更友好的。

案例

需求:给/order/{orderId}这个资源设置限流,最大QPS为10,利用排队的流控效果,超时时长设置为5s

实现

1)添加流控规则

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qdRoZ3fJ-1671787346301)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222221941894.png)]

2)Jmeter测试

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-kPCaUcCh-1671787346302)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222222028142.png)]

而我们设置的qps为10,我们查看结果树:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-TQhZVBWc-1671787346302)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222222104045.png)]

虽然我们实际的qps(15)大于了我们限定的qps(10),但是请求基本上都是成功的。

再去sentinel查看实时监控的QPS曲线:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-sAjS1pJW-1671787346302)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716114522935.png)]

QPS非常平滑,一致保持在10,但是超出的请求没有被拒绝,而是放入队列。因此响应时间(等待时间)会越来越长。

当队列满了以后,才会有部分请求失败:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-itT4LYzX-1671787346303)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716114651137.png)]


2.4.3 总结

流控效果有哪些?

  • 快速失败:QPS超过阈值时,拒绝新的请求

  • warm up: QPS超过阈值时,拒绝新的请求;QPS阈值是逐渐提升的,可以避免冷启动时高并发导致服务宕机。

  • 排队等待:请求会进入队列,按照阈值允许的时间间隔依次执行请求;如果请求预期等待时长大于超时时间,直接拒绝


2.5 热点参数限流

之前的限流是统计访问某个资源的所有请求,判断是否超过QPS阈值。而热点参数限流是分别统计参数值相同的请求,判断是否超过QPS阈值。

2.5.1 全局参数限流

例如,一个根据id查询商品的接口:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gIlGEQ1f-1671787346303)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716115014663.png)]

访问/goods/{id}的请求中,id参数值会有变化,热点参数限流会根据参数值分别统计QPS,统计结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-1RLHH2AX-1671787346303)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716115131463.png)]

当id=1的请求触发阈值被限流时,id值不为1的请求不受影响。

配置示例:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-YLrNfxI0-1671787346304)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716115232426.png)]

代表的含义是:对hot这个资源的0号参数(第一个参数)做统计,每1秒相同参数值的请求数不能超过5


2.5.2 快速入门

刚才的配置中,对查询商品这个接口的所有商品一视同仁,QPS都限定为5.

而在实际开发中,可能部分商品是热点商品,例如秒杀商品,我们希望这部分商品的QPS限制与其它商品不一样,高一些。那就需要配置热点参数限流的高级选项了:
在这里插入图片描述

结合上一个配置,这里的含义是对0号的long类型参数限流,每1秒相同参数的QPS不能超过5,有两个例外:

•如果参数值是100,则每1秒允许的QPS为10

•如果参数值是101,则每1秒允许的QPS为15

实战案例

需求

给/order/{orderId}这个资源添加热点参数限流,规则如下:

•默认的热点参数规则是每1秒请求量不超过2

•给102这个参数设置例外:每1秒请求量不超过4

•给103这个参数设置例外:每1秒请求量不超过10

**注意:**热点参数限流对默认的SpringMVC资源无效,需要利用 @SentinelResource 注解标记资源

步骤

1)标记资源

给order-service中的OrderController中的/order/{orderId}资源添加注解:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-udYTn8aK-1671787346305)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210716120033572.png)]

2)热点参数限流规则,点击左侧菜单的"热点规则":

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-FDS3mzMi-1671787346305)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210716120319009.png)]

填写表单如下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fJ4VKBam-1671787346305)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222231122389.png)]

3)Jmeter测试

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fDaoHmqJ-1671787346306)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222231207285.png)]

这里包含三个http请求。如图:

普通参数,qps阈值为2.

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-WLljzlhr-1671787346306)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222231247028.png)]

运行结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-aA7xmqFl-1671787346307)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222231525596.png)]

例外项,qps阈值为4.

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Ohku7AZu-1671787346307)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222231257501.png)]

运行结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-QNTnPv8A-1671787346307)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222231632483.png)]

例外项,qps为10.

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fEqV0svX-1671787346308)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222231308404.png)]

运行结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-V70bxxeT-1671787346308)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221222231720740.png)]


3. 隔离和降级

限流是一种预防措施,虽然限流可以尽量避免因高并发而引起的服务故障,但服务还会因为其它原因而故障。

而要将这些故障控制在一定范围,避免雪崩,就要靠线程隔离(舱壁模式)和熔断降级手段了。

线程隔离之前讲到过:调用者在调用服务提供者时,给每个调用的请求分配独立线程池,出现故障时,最多消耗这个线程池内资源,避免把调用者的所有资源耗尽。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-uicl603F-1671787346309)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210715173215243.png)]

熔断降级:是在调用方这边加入断路器,统计对服务提供者的调用,如果调用的失败比例过高,则熔断该业务,不允许访问该服务的提供者了。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5aSRaqJ9-1671787346309)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210715173428073.png)]

可以看到,不管是线程隔离还是熔断降级,都是对客户端(调用方)的保护。需要在调用方 发起远程调用时做线程隔离、或者服务熔断。

而我们的微服务远程调用都是基于Feign来完成的,因此我们需要将Feign与Sentinel整合,在Feign里面实现线程隔离和服务熔断。


3.1 降级

FeignClient整合Sentinel

SpringCloud中,微服务调用都是通过Feign来实现的,因此做客户端保护必须整合Feign和Sentinel。

3.1.1 修改配置,开启sentinel功能

修改order-service的配置文件,开启feign的sentinel功能:

feign:
  sentinel:
    enabled: true # 开启feign对sentinel的支持

3.1.2 编写失败降级逻辑

业务失败后,不能直接报错,而应该返回用户一个友好提示或者默认结果,这个就是失败降级逻辑。

给FeignClient编写失败后的降级逻辑有两种方式:

  1. FallbackClass,无法对远程调用的异常做处理
  2. FallbackFactory,可以对远程调用的异常做处理,我们选择这种

我们演示方式二的逻辑:

1)在feign-api项目中定义类,实现 FallbackFactory 接口:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-WoKT7bFH-1671787346310)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223113821740.png)]

代码如下:

@Slf4j
public class UserClientFallbackFactory implements FallbackFactory<UserClient> {
    @Override
    public UserClient create(Throwable throwable) {
        return new UserClient() {
            @Override
            public User findById(Long id) {
                log.error("查询用户异常",throwable);
                return new User();
            }
        };
    }
}

2)在feing-api项目中的DefaultFeignConfiguration类(该项目中的任意配置类都可)中将UserClientFallbackFactory注册为一个Bean:

@Bean
public UserClientFallbackFactory userClientFallbackFactory(){
    return new UserClientFallbackFactory();
}

3)在feing-api项目中的UserClient接口中使用UserClientFallbackFactory:

@FeignClient(value = "userservice",fallbackFactory = UserClientFallbackFactory.class)
public interface UserClient {

    @GetMapping("/user/{id}")
    User findById(@PathVariable("id") Long id);
}

4)重启后,访问该接口然后去查看sentinel控制台,可以看到新的簇点链路:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-NA9FKsET-1671787346310)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223120349022.png)]


3.1.3 总结

Sentinel支持的雪崩解决方案:

  • 线程隔离(舱壁模式)
  • 降级熔断

Feign整合Sentinel的步骤:

  • 在application.yml中配置:feign.sentienl.enable=true
  • 给FeignClient编写FallbackFactory并注册为Bean
  • 将FallbackFactory配置到FeignClient

3.2 隔离(舱壁模式)

3.2.1 线程隔离的实现方式

线程隔离有两种实现方式:

  • 线程池隔离
  • 信号量隔离(Sentinel默认采用)

如图:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tTFa6E5u-1671787346310)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716123036937.png)]

线程池隔离:给每个服务调用业务分配一个线程池,利用线程池本身实现隔离效果

信号量隔离:不创建线程池,而是计数器模式,记录业务使用的线程数量,达到信号量上限时,禁止新的请求。

两者的优缺点:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-kMSG9zkw-1671787346311)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716123240518.png)]


3.2.2 Sentinel的线程隔离

添加限流规则时,可以选择两种阈值类型:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-osPXwrOd-1671787346311)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210716123411217.png)]

  • QPS:就是每秒的请求数,在快速入门中已经演示过。
  • 线程数:是该资源能使用的tomcat线程数的最大值。也就是通过限制线程数量,实现线程隔离(舱壁模式)。

案例需求

给 order-service服务中的UserClient的查询用户接口设置流控规则,线程数不能超过 2。然后利用jemeter测试。

步骤

1)配置隔离规则

选择feign接口后面的流控按钮:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-daMjZ6gf-1671787346312)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716123831992.png)]

填写表单:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5zJkeRB0-1671787346312)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716123936844.png)]

2)Jmeter测试

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-e1STi5GZ-1671787346312)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223152808795.png)]

一次发生10个请求,有较大概率并发线程数超过2,而超出的请求会走之前定义的失败降级逻辑。

查看运行结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-dW2EEuLm-1671787346313)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223152900995.png)]

虽然结果都是成功的,不过部分请求得到的响应结果是降级返回的null信息。


3.2.3 总结

线程隔离的两种手段:

  • 信号量隔离
  • 线程池隔离

信号量隔离的特点是?

基于计数器模式,简单,开销小

线程池隔离的特点是?

基于线程池模式,有额外开销,但隔离控制更强


3.3 熔断降级

熔断降级是解决雪崩问题的重要手段。其思路是由断路器统计服务调用的异常比例、慢请求比例,如果超出阈值则会熔断该服务。即拦截访问该服务的一切请求;而当服务恢复时,断路器会放行访问该服务的请求。

断路器控制熔断和放行是通过状态机来完成的:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-8H0CoAwo-1671787346313)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716130958518.png)]

状态机包括三个状态:

  • closed:关闭状态,断路器放行所有请求,并开始统计异常比例、慢请求比例。超过阈值则切换到open状态
  • open:打开状态,服务调用被熔断,访问被熔断服务的请求会被拒绝,快速失败,直接走降级逻辑。Open状态5秒后会进入half-open状态
  • half-open:半开状态,放行一次请求,根据执行结果来判断接下来的操作。
    • 请求成功:则切换到closed状态
    • 请求失败:则切换到open状态

断路器熔断策略有三种:慢调用、异常比例、异常数


3.3.1 慢调用

慢调用:业务的响应时长(RT)大于指定时长的请求认定为慢调用请求。在指定时间内,如果请求数量超过设定的最小数量,慢调用比例大于设定的阈值,则触发熔断。

例如:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-V2dgEFwA-1671787346314)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716145934347.png)]

解读:RT超过500ms的调用是慢调用,统计最近10000ms内的请求,如果请求量超过10次,并且慢调用比例不低于0.5,则触发熔断,熔断时长为5秒。然后进入half-open状态,放行一次请求做测试。

案例

需求:给 UserClient的查询用户接口设置降级规则,慢调用的RT阈值为50ms,统计时间为1秒,最小请求数量为5,失败阈值比例为0.4,熔断时长为5

步骤

1)设置慢调用

修改user-service中的/user/{id}这个接口的业务。通过休眠模拟一个延迟时间:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-b6VeGb2m-1671787346314)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223160915066.png)]

2)此时,orderId==101的订单关联的用户id为1,调用时长超过了50ms:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-2heiAT4V-1671787346315)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223161020801.png)]

而orderId==102的订单因为关联的用户id不是1,调用市场就短了很多:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-e5oTvCxS-1671787346315)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223161117144.png)]

3)设置熔断规则

下面,给feign接口设置降级规则:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-NOyE9ekd-1671787346316)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716150654094.png)]

规则:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-RYH9DxGk-1671787346316)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716150740434.png)]

超过50ms的请求都会被认为是慢请求

4)测试

在浏览器访问:http://localhost:8088/order/101,快速刷新5次,可以发现:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-pIwtTOfg-1671787346316)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223161323643.png)]

触发了熔断,请求时长缩短至5ms,快速失败了,并且走降级逻辑,返回的null

在浏览器访问:http://localhost:8088/order/102,竟然也被熔断了:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-EuKt9xN5-1671787346317)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223161427145.png)]


3.3.2 异常比例、异常数

异常比例或异常数:统计指定时间内的调用,如果调用次数超过指定请求数,并且出现异常的比例达到设定的比例阈值(或超过指定异常数),则触发熔断。

例如,一个异常比例设置:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-HIbN9v11-1671787346317)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716131430682.png)]

解读:统计最近1000ms内的请求,如果请求量超过10次,并且异常比例不低于0.4,则触发熔断。

一个异常数设置:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5brMM8mC-1671787346317)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716131522912.png)]

解读:统计最近1000ms内的请求,如果请求量超过10次,并且异常比例不低于2次,则触发熔断。

案例

需求:给UserClient的查询用户接口设置降级规则,统计时间为1秒,最小请求数量为5,失败阈值比例为0.4,熔断时长为5s

1)设置异常请求

首先,修改user-service中的/user/{id}这个接口的业务。手动抛出异常,以触发异常比例的熔断:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-xFBDAhnh-1671787346318)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223162544556.png)]

3)设置熔断规则

下面,给feign接口设置降级规则:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-BTYh7H5y-1671787346318)(D:\学习笔记\Java\黑马\SpringCloud\高级篇\day01-微服务保护\讲义\assets\image-20210716151538785.png)]

在5次请求中,只要异常比例超过0.4,也就是有2次以上的异常,就会触发熔断。

4)测试

在浏览器快速访问:http://localhost:8088/order/102,快速刷新5次,触发熔断:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-B2ErYlvR-1671787346318)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716151722916.png)]

此时,我们去访问本来应该正常的103:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-HdwgrDqx-1671787346319)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716151844817.png)]


4. 授权

授权规则可以对请求方来源做判断和控制。

4.1 授权规则

4.1.1 基本规则

授权规则可以对调用方的来源做控制,有白名单和黑名单两种方式。

  • 白名单:来源(origin)在白名单内的调用者允许访问
  • 黑名单:来源(origin)在黑名单内的调用者不允许访问

点击左侧菜单的授权,可以看到授权规则:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-g3o6GRZ5-1671787346319)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716152010750.png)]

  • 资源名:就是受保护的资源,例如/order/{orderId}

  • 流控应用:是来源者的名单,

    • 如果是勾选白名单,则名单中的来源被许可访问。
    • 如果是勾选黑名单,则名单中的来源被禁止访问。

比如:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-MqU81KRO-1671787346320)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716152349191.png)]

我们允许请求从gateway到order-service,不允许浏览器访问order-service,那么白名单中就要填写网关的来源名称(origin)


4.1.2 如何获取origin

Sentinel是通过RequestOriginParser这个接口的parseOrigin来获取请求的来源的。

public interface RequestOriginParser {
    /**
     * 从请求request对象中获取origin,获取方式自定义
     */
    String parseOrigin(HttpServletRequest request);
}

这个方法的作用就是从request对象中,获取请求者的origin值并返回。

默认情况下,sentinel不管请求者从哪里来,返回值永远是default,也就是说一切请求的来源都被认为是一样的值default。

因此,我们需要自定义这个接口的实现,让不同的请求,返回不同的origin

例如order-service服务中,我们定义一个RequestOriginParser的实现类:

@Component
public class HeaderOriginParser implements RequestOriginParser {
    @Override
    public String parseOrigin(HttpServletRequest httpServletRequest) {
        //1.获取请求头
        String origin = httpServletRequest.getHeader("origin");
        if(StringUtils.isEmpty(origin)){
            return "blank";
        }
        return origin;
    }
}

我们会尝试从request-header中获取origin值。


4.1.3 给网关添加请求头

既然获取请求origin的方式是从reques-header中获取origin值,我们必须让所有从gateway路由到微服务的请求都带上origin头

这个需要利用之前学习的一个GatewayFilter来实现,AddRequestHeaderGatewayFilter。

修改gateway服务中的application.yml,添加一个defaultFilter:

spring:
  cloud:
    gateway:
      default-filters:
        - AddRequestHeader=origin,gateway
      routes:
       # ...略

这样,从gateway路由的所有请求都会带上origin头,值为gateway。而从其它地方到达微服务的请求则没有这个头。


4.1.4 配置授权规则

接下来,我们添加一个授权规则,放行origin值为gateway的请求。

在这里插入图片描述

配置如下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-K5n6wvh9-1671787346320)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716153301069.png)]

现在,我们直接跳过网关,访问order-service服务,被拦截:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-seOoLR9w-1671787346321)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223164836526.png)]

通过网关访问,访问成功:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6BjwSB8B-1671787346321)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223164856270.png)]


4.2 自定义异常结果

默认情况下,发生限流、降级、授权拦截时,都会抛出异常到调用方。异常结果都是flow limmiting(限流)。这样不够友好,无法得知是限流还是降级还是授权拦截。

4.2.1 异常类型

而如果要自定义异常时的返回结果,需要实现BlockExceptionHandler接口:

public interface BlockExceptionHandler {
    /**
     * 处理请求被限流、降级、授权拦截时抛出的异常:BlockException
     */
    void handle(HttpServletRequest request, HttpServletResponse response, BlockException e) throws Exception;
}

这个方法有三个参数:

  • HttpServletRequest request:request对象
  • HttpServletResponse response:response对象
  • BlockException e:被sentinel拦截时抛出的异常

这里的BlockException包含多个不同的子类:

异常说明
FlowException限流异常
ParamFlowException热点参数限流的异常
DegradeException降级异常
AuthorityException授权规则异常
SystemBlockException系统规则异常

4.2.2 自定义异常处理

下面,我们就在order-service定义一个自定义异常处理类:

@Component
public class SentinelExceptionHandler implements BlockExceptionHandler {
    @Override
    public void handle(HttpServletRequest request, HttpServletResponse response, BlockException e) throws Exception {
        String msg = "未知异常";
        int status = 429;

        if (e instanceof FlowException) {
            msg = "请求被限流了";
        } else if (e instanceof ParamFlowException) {
            msg = "请求被热点参数限流";
        } else if (e instanceof DegradeException) {
            msg = "请求被降级了";
        } else if (e instanceof AuthorityException) {
            msg = "没有权限访问";
            status = 401;
        }

        response.setContentType("application/json;charset=utf-8");
        response.setStatus(status);
        response.getWriter().println("{\"msg\": " + msg + ", \"status\": " + status + "}");
    }
}

重启测试,在不同长津峡,会返回不同的异常信息。

限流:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-IDoQtnbh-1671787346322)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223165841425.png)]

授权拦截时:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-pyTR4nhx-1671788089271)(C:\Users\zhuhuanjie\AppData\Roaming\Typora\typora-user-images\image-20221223165741037.png)]


4.2.3 总结

获取请求来源的接口是什么(白名单)?

  • RequestOriginParser

处理BlockException的接口是什么?

  • BlockExceptionHandler

5. 规则持久化

现在,sentinel的所有规则都是内存存储,重启后所有规则都会丢失。在生产环境下,我们必须确保这些规则的持久化,避免丢失。

5.1 规则管理模式

规则是否能够持久化,取决于规则管理模式,sentinel支持三种规则管理模式:

  • 原始模式:Sentinel的默认模式,将规则保存在内存,重启服务会丢失。
  • pull模式
  • push模式

5.1.1 pull模式

pull模式:控制台将配置的规则推送到Sentinel客户端,而客户端会将配置规则保存在本地文件或数据库中。以后会定时去本地文件或数据库中查询,更新本地规则。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tUOsQbbP-1671787346323)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716154155238.png)]


5.1.2 push模式

push模式:控制台将配置规则推送到远程配置中心,例如Nacos。Sentinel客户端监听Nacos,获取配置变更的推送消息,完成本地配置更新。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-dX6RWcFa-1671787346323)(D:/学习笔记/Java/黑马/SpringCloud/高级篇/day01-微服务保护/讲义/assets/image-20210716154215456.png)]

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值