什么是服务雪崩
在微服务调用链路中,因服务提供者的不可用导致服务调用者的不可用,并将不可用逐渐放大的过程,就叫服务雪崩效应。
导致服务不可用的原因:
data:image/s3,"s3://crabby-images/fcfac/fcfaca8c1ed8d95a68334383981278da975844af" alt=""
解决方案
1、超时机制
data:image/s3,"s3://crabby-images/dec45/dec454273c3e579bd980796a7b222a7b4686bea7" alt=""
2、服务限流
3、舱壁隔离(资源隔离)
4、服务熔断降级
data:image/s3,"s3://crabby-images/d14ce/d14ce314c8d68028b758a13311aa19e08206a730" alt=""
data:image/s3,"s3://crabby-images/3300c/3300c6e70274e8cd79ec469785f07bd15f47db12" alt=""
技术选型: Sentinel or Hystrix
data:image/s3,"s3://crabby-images/314d3/314d32e039169db2e18444d67a56336a21e07c5e" alt=""
对比
data:image/s3,"s3://crabby-images/d543e/d543e58c11516202c4fd4f456f488349e7f469eb" alt=""
Sentinel快速开始
data:image/s3,"s3://crabby-images/97dc7/97dc724b2168175c2bc226000a9e0d6bdeeeb969" alt=""
spring boot 微服务 应用接入Sentinel控制台
<!--引入依赖 sentinel -->
<!--引入依赖 sentinel --> <dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-starter-alibaba-sentinel</artifactId> </dependency>
配置 application.yml 文件
server:
port: 8800
#开启Sentinel 对 Feign 的支持
feign:
sentinel:
enabled: true
spring:
application:
name: mall-user-sentinel-demo
cloud:
nacos:
discovery:
server-addr: 192.168.56.1:8848
username: nacos
password: nacos
openfeign: #openfeign的配置
client:
config:
mall-order: # 对应微服务
loggerLevel: FULL
# 连接超时时间
connectTimeout: 3000
# 请求处理超时时间
readTimeout: 5000
okhttp:
enabled: true
sentinel:
transport:
dashboard: 192.168.116.156:8858
# datasource:
# flow-rules:
# nacos:
# server-addr: 192.168.56.1:8848
# namespace: xulk
# username: nacos
# password: nacos
# dataId: ${spring.application.name}-flow-rules
# groupId: SENTINEL_GROUP
# data-type: json
# rule-type: flow
datasource:
type: com.alibaba.druid.pool.DruidDataSource
driver-class-name: com.mysql.cj.jdbc.Driver
url: jdbc:mysql://localhost:3306/ms_user?useUnicode=true&characterEncoding=UTF-8&serverTimezone=Asia/Shanghai
username: root
password: root
druid:
initial-size: 10
max-active: 100
min-idle: 10
max-wait: 60000
pool-prepared-statements: true
max-pool-prepared-statement-per-connection-size: 20
time-between-eviction-runs-millis: 60000
min-evictable-idle-time-millis: 300000
test-while-idle: true
test-on-borrow: false
test-on-return: false
stat-view-servlet:
enabled: true
url-pattern: /druid/*
filter:
stat:
log-slow-sql: true
slow-sql-millis: 1000
merge-sql: false
wall:
config:
multi-statement-allow: true
#暴露actuator端点 http://localhost:8800/actuator/sentinel
management:
endpoints:
web:
exposure:
include: '*'
logging:
level:
com.tuling.mall.sentineldemo.feign: debug
com.alibaba.cloud.nacos.client.NacosPropertySourceBuilder: debug
controller 类
@RequestMapping(value = "/findOrderByUserId/{id}")
//@SentinelResource(value = "findOrderByUserId",blockHandler = "handleException")
public R findOrderByUserId(@PathVariable("id") Integer id) {
return R.ok( "11111111111");
}
远程调用 feign 接口
import com.tuling.common.utils.R;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
@FeignClient(value = "mall-order",path = "/order",fallbackFactory = FallbackOrderFeignServiceFactory.class)
public interface OrderFeignService {
@RequestMapping("/findOrderByUserId/{userId}")
public R findOrderByUserId(@PathVariable("userId") Integer userId);
}
进入降级类
import com.tuling.common.utils.R;
import com.tuling.mall.sentineldemo.feign.OrderFeignService;
import org.springframework.cloud.openfeign.FallbackFactory;
import org.springframework.stereotype.Component;
@Component
public class FallbackOrderFeignServiceFactory implements FallbackFactory<OrderFeignService> {
@Override
public OrderFeignService create(Throwable throwable) {
return new OrderFeignService() {
@Override
public R findOrderByUserId(Integer userId) {
return R.error(-1,"=======服务降级了5555555========");
}
};
}
}
自定义异常 MyBlockExceptionHandler
mport com.alibaba.csp.sentinel.adapter.spring.webmvc.callback.BlockExceptionHandler;
import com.alibaba.csp.sentinel.slots.block.BlockException;
import com.alibaba.csp.sentinel.slots.block.authority.AuthorityException;
import com.alibaba.csp.sentinel.slots.block.degrade.DegradeException;
import com.alibaba.csp.sentinel.slots.block.flow.FlowException;
import com.alibaba.csp.sentinel.slots.block.flow.param.ParamFlowException;
import com.alibaba.csp.sentinel.slots.system.SystemBlockException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.tuling.common.utils.R;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import lombok.extern.slf4j.Slf4j;
import org.springframework.http.MediaType;
import org.springframework.stereotype.Component;
import java.io.PrintWriter;
@Slf4j
@Component
public class MyBlockExceptionHandler implements BlockExceptionHandler {
@Override
public void handle(HttpServletRequest request, HttpServletResponse response, BlockException e) throws Exception {
log.info("BlockExceptionHandler BlockException================"+e.getRule());
R r = null;
if (e instanceof FlowException) {
r = R.error(100,"接口限流了111");
} else if (e instanceof DegradeException) {
r = R.error(101,"服务降级了2222");
} else if (e instanceof ParamFlowException) {
r = R.error(102,"热点参数限流了3333");
} else if (e instanceof SystemBlockException) {
r = R.error(103,"触发系统保护规则了444");
} else if (e instanceof AuthorityException) {
r = R.error(104,"授权规则不通过5555");
}
//返回json数据
response.setStatus(500);
response.setCharacterEncoding("utf-8");
response.setContentType(MediaType.APPLICATION_JSON_VALUE);
new ObjectMapper().writeValue(response.getWriter(), r);
}
}
配置流控规则 默认没持久化,重启服务 后会丢失
data:image/s3,"s3://crabby-images/26b14/26b1427b01fb89e4b954df734573e8ee29330c6a" alt=""
data:image/s3,"s3://crabby-images/6d39b/6d39bf3e584b617c3ecb4ab3768144e4ffab6504" alt=""
测试 http://localhost:8800/user/findOrderByUserId/1
data:image/s3,"s3://crabby-images/5bea6/5bea6e5d4dddb969e54447c03e98fc04ad3fba36" alt=""
data:image/s3,"s3://crabby-images/5e8c0/5e8c0cad2fd085f5b678757b9531177af73a6785" alt=""
降级/熔断处理配置
data:image/s3,"s3://crabby-images/58723/58723a8b28dcd4278cc293391c471ebe5879e231" alt=""
data:image/s3,"s3://crabby-images/efb68/efb6832f466d457c8ea92447cb24013aa2181485" alt=""
sentinel 持久化
<!--sentinel 从nacos拉取规则 --> <dependency> <groupId>com.alibaba.csp</groupId> <artifactId>sentinel-datasource-nacos</artifactId> </dependency>
配置文件 添加nacos的地址和配置文件
data:image/s3,"s3://crabby-images/a6461/a646193fef3879d765922cecabffafb9a599e252" alt=""
data:image/s3,"s3://crabby-images/14646/146464c199784fabc4c36a33ffca1e59d7cfb056" alt=""
访问 http://localhost:8800/user/findOrderByUserId/1
data:image/s3,"s3://crabby-images/5f319/5f319254bbb0dc401ae5f57a21885054b9a10b13" alt=""
自动持久化到 nacos
data:image/s3,"s3://crabby-images/40d2c/40d2ce2d307493d80060a2c0c4555a79ff425441" alt=""