在微服务架构中,根据业务来拆分成一个个的服务,服务与服务之间可以通过RPC相互调用,在Spring Cloud中可以用RestTemplate + Ribbon和Feign来调用,为了保证其高可用,单个服务通常会集群部署,由于网络原因或者自身的原因,服务并不能保证100%可用,如果单个服务出现问题,调用这个服务就会线程阻塞,此时若有大量请求涌入,Servlet容器的线程资源会被消耗完毕,导致服务瘫痪,服务与服务之间的依赖性,故障会传播,会对整个微服务系统造成灾难性的严重后果,这就是服务雪崩效应。
设计原则
1、防止单个服务的故障耗尽整个服务的servlet容器的线程资源
2、快速失败机制,如果某个服务出现故障则调用该服务的请求快速失败,而不是线程等待
3、提供回退方案
4、使用熔断机制,防止故障扩散到其他服务
5、提供熔断器的监控组件Hystrix Dashboard,实时监控熔断器状态。提供Turbine聚合多喝Dashboard
工作机制
1、当某个API接口失败的次数在一定时间内小鱼设定的阀值时,熔断器处于关闭状态,该API正常提供服务。
2、当失败次数大于设定的阀值的时候,Hystrix判定改API接口出现故障,打开熔断器,这时候该请求API接口会执行快速失败的逻辑(fallback回退的逻辑)而不执行业务逻辑,请求的线程不会处于阻塞状态。
3、处于打开状态的熔断器,一段时间后会处于半打开状态,并将一定数量的请求执行业务逻辑,剩余的请求会执行快速失败。若执行的业务逻辑请求失败,则熔断器继续打开,若成功则熔断器关闭。
熔断器可以在RestTemplate+Ribbon 或 feign 服务中使用
Ribbon中使用Hystrix(熔断器)
在Ribbon的pom.xml中添加依赖
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-hystrix</artifactId>
</dependency>
2. 在application中添加注解@EnableHystrix启动熔断器Hystrix
package com.funtl.hello.spring.cloud.web.admin.ribbon;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
import org.springframework.cloud.netflix.hystrix.EnableHystrix;
@EnableHystrix
@SpringBootApplication
@EnableDiscoveryClient
public class WebAdminRibbonApplication {
public static void main(String[] args) {
SpringApplication.run(WebAdminRibbonApplication.class,args);
}
}
3.Ribbon 调用方法上增加 @HystrixCommand 注解并指定 fallbackMethod 熔断方法
package com.funtl.hello.spring.cloud.web.admin.ribbon.service;
import com.netflix.hystrix.contrib.javanica.annotation.HystrixCommand;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.web.client.RestTemplate;
@Service
public class AdminService {
@Autowired
private RestTemplate restTemplate;
@HystrixCommand(fallbackMethod = "hiError")
public String sayHi(String message) {
return restTemplate.getForObject("http://HELLO-SPRING-CLOUD-SERVICE-ADMIN/hi?message=" + message, String.class);
}
public String hiError(String message){
return String.format("Hi your message is : %s but request bad",message);
}
}
测试熔断器
此时关闭关闭服务提供者,再次请求http://localhost:8764/hi?messsage=HelloRibbon 浏览器会显示
Hi,your message is :"HelloRibbon" but request error."
Feign使用熔断器Hystrix
feign自带Hystrix但是默认是关闭的,需要手动启动
在application.yml中添加
feign:
hystrix:
enabled: true
2. 在service中添加fallback指定类
package com.funtl.hello.spring.cloud.web.admin.feign.service;
import com.funtl.hello.spring.cloud.web.admin.feign.service.hystrix.AdminSeviceHystrix;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;
@FeignClient(value = "hello-spring-cloud-service-admin",fallback = AdminSeviceHystrix.class)
public interface AdminService {
@RequestMapping(value = "hi",method = RequestMethod.GET)
public String sayHi(@RequestParam(value = "message") String message);
}
创建熔断器类并实现对应的Feign接口
package com.funtl.hello.spring.cloud.web.admin.feign.service.hystrix;
import com.funtl.hello.spring.cloud.web.admin.feign.service.AdminService;
import org.springframework.stereotype.Component;
@Component
public class AdminSeviceHystrix implements AdminService {
@Override
public String sayHi(String message) {
return "Hi,your message is :\"" + message + "\"but request error.";
}
}
测试熔断器
此时关闭关闭服务提供者,再次请求http://localhost:8764/hi?messsage=HelloFeign 浏览器会显
Hi,your message is :"HelloFeign" but request error."
往期文章:SpringCloud入门(一)-springcloud简介与统一版本依赖管理
SpringCloud入门(二)-服务注册与发现
SpringCloud入门(三)-服务提供者