继续接着上一篇的项目来整合Fegin,Hystrix,Rule。上一篇地址 https://blog.csdn.net/qq_36882793/article/details/96301559
项目地址: https://download.csdn.net/download/qq_36882793/11389879
Feign可以把Rest的请求进行隐藏,伪装成类似SpringMVC的Controller一样。你不用再自己拼接url,拼接参数等等操作,一切都交给Feign去做。
项目主页:https://github.com/OpenFeign/feign
即在之前的项目中,我们调用dirver服务是通过RestTesmplate来通过拼接url来实现服务调用,并不合适。通过Fegin我们可以简化这种操作。
org.springframework.cloud
spring-cloud-starter-openfeign
在乘客服务启动类中添加 @EnableFeignClients 注解 启动Fegin客户端。另外@SpringBootApplication这里的注解换成了@SpringCloudApplication,这个注解集成了@SpringBootApplication、@EnableDiscoveryClient、@EnableCircuitBreaker 注解。
我们新建一个client文件夹,创建接口PassengerClient。在接口上添加注解@FeignClient 标志该类是一个fegin接口,value值是需要跳转的服务的服务名(即ApplicationName)。可以看到PassengerController中的callTaxi 方法调用的是eurake-dirver服务中的taskOrders方法,所以我们要在PassengerClient声明一个taskOrders 接口来代理该方法。
如下图所示,这里通过@RequestMappering 来指定调用的路径为eurake-driver服务的driver/taksOrders 路径。这里的taskOrders接口方法和 eurake-driver中的taskOrders方法在定义上在返回值相同,参数列表相同,
回到PassengerController 中,Fegin 中已经集成了RestTemplate, 所以无需我们自己再注入。也集成Ribbon负载均衡,并且默认启动了负载均衡。所以我们也不用引入和集成Ribbon包。这里注入PassengerClient,并通过他来调用司机的服务。
Hystrix,英文意思是豪猪,全身是刺,看起来就不好惹,是一种保护机制。
Hystrix也是Netflix公司的一款组件。
主页:https://github.com/Netflix/Hystrix/
熔断状态机3个状态:
Feign默认也有对Hystrix的集成,但是默认情况下是关闭的。
feign:
hystrix:
enabled: true # 开启熔断功能
yml文件
server:
port: 10005
eureka:
client:
service-url: # 注册到服务中心 (滴滴平台)
defaultZone: http://127.0.0.1:10000/eureka
instance: # 使用自定义ip
preferIpAddress: true
instance-id: ${spring.application.name}:${server.port}
spring:
application: # 应用名称,在服务中心显示
name: eureka-zuul
zuul:
routes:
service-provider: # 路由id,可以随便
path: /eurake_passenger/** # 这里是映射路径 --- 即 访问时的地址。** 代表所有
url: http://127.0.0.1:10001 # 映射路径对应的实际url地址 --- 即真实访问的服务地址
但是上述我们将url地址写死为127.0.0.1,当服务迁移时则无法使用,所以我们需要建立面向服务的映射关系。说来也简单。。。。。将zull的yml文件url改为对应的服务名称即可。同时也会启用负载均衡
简化路由配置:
上面的routes配置可以简化为如下,也是没人配置,这时routes的配置key值是服务名称,value值是映射路径:
zuul:
routes:
eureka-passenger : /eurake_passenger/** # 这里是映射路径
public abstract ZuulFilter implements IZuulFilter{
abstract public String filterType(); //返回字符串,代表过滤器的类型。包含以下4种: pre:请求在被路由之前执行 route:在路由请求时调用; post:在route和errror过滤器之后调用; error:处理请求时发生错误调用
abstract public int filterOrder(); // 通过返回的int值来定义过滤器的执行顺序,数字越小优先级越高
boolean shouldFilter();// 返回一个Boolean值,判断该过滤器是否需要执行。返回true执行,返回false不执行
Object run() throws ZuulException;// 过滤器的具体业务逻辑
}
正常流程:
异常流程:
package com.kingfish.filter;
import com.netflix.zuul.ZuulFilter;
import com.netflix.zuul.context.RequestContext;
import com.netflix.zuul.exception.ZuulException;
import org.apache.http.HttpStatus;
import org.springframework.stereotype.Component;
import javax.servlet.http.HttpServletRequest;
/**
* @Data: 2019/7/20
* @Des: 自定义zuul过滤器
*/
@Component
public class MyFilter extends ZuulFilter {
/**
* 过滤器类型,这里设置为前置过滤器
*
* @return
*/
@Override
public String filterType() {
return "pre";
}
/**
* 过滤器的执行顺序
*
* @return
*/
@Override
public int filterOrder() {
return 1;
}
/**
* 该过滤器是否生效
* @return
*/
@Override
public boolean shouldFilter() {
return true;
}
/**
* 过滤逻辑
* @return
* @throws ZuulException
*/
@Override
public Object run() throws ZuulException {
// 获取zuul提供的上下文对象
RequestContext context = RequestContext.getCurrentContext();
// 从上下文对象中获取请求对象
HttpServletRequest request = context.getRequest();
// 获取name信息
String name = request.getParameter("name");
// 判断
if ("李四".equals(name)) {
// 过滤该请求,不对其进行路由
context.setSendZuulResponse(false);
// 设置响应状态码,401
context.setResponseStatusCode(HttpStatus.SC_UNAUTHORIZED);
// 设置响应信息
context.setResponseBody("{\"status\":\"401\", \"text\":\"request error!\"}");
}
// 校验通过,把name信息放入上下文信息,继续向后执行
context.set("name", name);
return null;
}
}
注: Zuul中默认就已经集成了Ribbon负载均衡和Hystix熔断机制。但是所有的超时策略都是走的默认值,比如熔断超时时间只有1秒,可通过以下配置进行配置
hystrix:
command:
default:
execution:
isolation:
thread:
timeoutInMilliseconds: 2000 # 设置hystrix的超时时间为6000ms