文章目录
前言
服务调用方案--Feign
声明式的Web服务客户端
一、Feign介绍
1. 什么是Feign
Feign是声明式的Web服务客户端,让编写Web服务客户端变得非常容易,只需创建一个接口并在接口上添加注解即可。
Feign 不做任何请求处理,通过处理注解相关信息生成 Request,并对调用返回的数据进行解码,从而实现 简化 HTTP API 的开发。
如果要使用 Feign,需要创建一个接口并对其添加 Feign 相关注解,另外 Feign 还支持可插拔编码器和解码器,致力于打造一个轻量级 HTTP 客户端。
2. 什么是Http客户端
HTTP(超文本传输协议)是一种应用层协议,用于客户端和服务端进行通信,按照标准格式如JSON、XML等进行网络数据的传输,通常也作为应用程序之间以REST API形式进行通信的常用协议。
在Java应用中需要调用其他应用提供的HTTP服务API时,通常需要使用一些HTTP客户端组件,对HTTP协议进行封装,将网络传输的功能转化为方法,开发人员就可以直接调用。
主要介绍的HTTP客户端包括:
-
Spring Boot中的
WebClient
:Java标准库的一部分。基于Java SE平台提供的HttpURLConnection类构建的。用于取代较旧的RestTemplate,以便在使用Spring Boot框架构建的应用程序中进行REST API调用,它支持同步、异步和流式处理。 -
Java 11+版本中提供的
HttpClient
:Java标准库的一部分,取代了JDK更早期的HttpUrlConnection类。与WebClient相比,HttpClient具有更好的性能和更多的功能。它支持连接池、重试机制、代理设置等高级特性。此外,HttpClient还提供了对HTTP/2和WebSocket的支持。 -
Apache HttpComponents项目中的
HttpClient
:可用于HTTP协议的Java工具集,HTTP代理实现。 -
OkHttpClient
:开源的HTTP客户端库。与WebClient和HttpClient相比,OkHttp更加灵活,易于扩展和定制。
Spring Boot 项目中,底层涉及网络请求的组件有 RestTemplate、Feign 和 Zuul,它们分别有自己默认的 HTTP 请求客户端,很多时候为了获得更好的性能,我们需要替换底层默认的 HTTP 客户端。
Feign 默认使用的是 JDK 原生的 HTTPURLConnection
。可以使用 Apache HTTP Client 或者 Okhttp 来进行替换,替换的步骤分为两步,首先引入相关的依赖库,然后修改配置。
3. Feign 和 OpenFeign 的区别
OpenFeign 组件的前身是 Netflix Feign 项目,它最早是作为 Netflix OSS 项目的一部分,由 Netflix 公司开发。后来 Feign 项目被贡献给了开源组织,于是才有了我们今天使用的 Spring Cloud OpenFeign 组件。
Spring Cloud 添加了对 Spring MVC 注解的支持,并支持使用 Spring Web 中默认使用的相同HttpMessageConverters。
另外,Spring Cloud 同时集成了 Ribbon 和 注册中心 (Eureka、Consul、Naocs等)以及 Spring Cloud LoadBalancer,以在使用 Feign 时提供负载均衡的 HTTP 客户端。
二、Feign底层原理
核心点围绕在动态代理,如何发送及接收 HTTP 网络请求
。
-
通过 @EnableFeignCleints 注解启动 Feign Starter 组件。
-
Feign Starter 在项目启动过程中注册全局配置,扫描包下所有的 @FeignClient 接口类,并进行注册 IOC 容器。
-
@FeignClient 接口类被注入时,通过 FactoryBean#getObject 返回动态代理类。创建动态代理类的方式和 Mybatis Mapper 处理方式是一致的,因为两者都没有实现类。
根据 newInstance 方法按照行为大致划分,共做了四件事:
- 处理 @FeignCLient 注解(SpringMvc 注解等)封装为 MethodHandler 包装类。
- 遍历接口中所有方法,过滤 Object 方法,并将默认方法以及 FeignClient 方法分类。
- 创建动态代理对应的 InvocationHandler 并创建 Proxy 实例。
- 接口内 default 方法 绑定动态代理类。
-
接口被调用时被动态代理类逻辑拦截,将 @FeignClient 请求信息通过编码器生成 HTTP Request。Feign 发送请求以及接收响应等都是由 Client 完成,该类默认 Client.Default,另外支持 HttpClient、OkHttp 等客户端。
-
交由 Ribbon 进行负载均衡,挑选出一个健康的 Server 实例。通过 Ribbon 获取服务列表,并对服务列表进行负载均衡调用(服务名转换为 ip+port)。
-
继而通过 Client 携带 Request 调用远端服务返回请求响应。
-
通过解码器生成 HTTP Response 返回客户端,将信息流解析成为接口返回数据。
三、Feign工作原理详解
OpenFeign 使用了动态代理技术来封装远程服务调用的过程,远程服务调用的信息被写在了被 @FeignClient 修饰的接口中。服务的名称、接口类型、访问路径已经通过注解做了声明。OpenFeign 通过解析这些注解标签生成一个“动态代理类”,这个代理类会将接口调用转化为一个远程服务调用的 Request,并发送给目标服务。
1. 动态代理机制
上图中的步骤 1 到步骤 3 是在项目启动阶段加载完成的,只有第 4 步“调用远程服务”是发生在项目的运行阶段。
-
在项目启动阶段,OpenFeign 框架会发起一个主动的扫包流程,从指定的目录下扫描并加载所有被 @FeignClient 注解修饰的接口。
-
OpenFeign 会针对每一个 FeignClient 接口生成一个动态代理对象,即图中的FeignProxyService,这个代理对象在继承关系上属于 FeignClient 注解所修饰的接口的实例。
-
这个动态代理对象会被添加到 Spring 上下文中,并注入到对应的服务里,也就是图中的 LocalService 服务。
-
LocalService 会发起底层方法调用。实际上这个方法调用会被 OpenFeign 生成的代理对象接管,由代理对象发起一个远程服务调用,并将调用的结果返回给LocalService。
总之,就是通过 Java 动态代理生成了一个“代理类”,这个代理类将接口调用转化成为了一个远程服务调用。
2. 动态代理的创建过程
如何通过动态代理技术创建代理对象的?
@EnableFeignClients,将修饰了 @FeignClient 的接口注册为 IOC Bean。
-
项目加载:在项目的启动阶段,EnableFeignClients 注解扮演了“启动开关”的角色,它使用 Spring 框架的 Import 注解导入了 FeignClientsRegistrar 类,开始了OpenFeign 组件的加载过程。
-
扫包:FeignClientsRegistrar 负责 FeignClient 接口的加载,它会在指定的包路径下扫描所有的 FeignClients 类,并构造 FeignClientFactoryBean 对象来解析FeignClient 接口。
-
解析 FeignClient 注解:FeignClientFactoryBean 有两个重要的功能,一个是解析FeignClient 接口中的请求路径和降级函数的配置信息;另一个是触发动态代理的构造过程。其中,动态代理构造是由更下一层的 ReflectiveFeign 完成的。
-
构建动态代理对象:ReflectiveFeign 包含了 OpenFeign 动态代理的核心逻辑,它主要负责创建出 FeignClient 接口的动态代理对象。
ReflectiveFeign 在这个过程中有两个重要任务:
-
解析 FeignClient 接口上各个方法级别的注解,将其中的远程接口URL、接口类型(GET、POST 等)、各个请求参数等封装成元数据,并为每一个方法生成一个对应的 MethodHandler 类作为方法级别的代理;
-
将这些MethodHandler 方法代理做进一步封装,通过 Java 标准的动态代理协议,构建一个实现了 InvocationHandler 接口的动态代理对象,并将这个动态代理对象绑定到FeignClient 接口上。这样一来,所有发生在 FeignClient 接口上的调用,最终都会由它背后的动态代理对象来承接。
其中,元数据的解析如何完成的呢?
依赖于 OpenFeign 组件中的Contract 协议解析功能。Contract 是 OpenFeign 组件中定义的顶层抽象接口,它有一系列的具体实现,其中和我们项目有关的是 SpringMvcContract 这个类。
SpringMvcContract 的继承结构是 SpringMvcContract->BaseContract->Contract。
详见OpenFeign 如何做到 “隔空取物”
3. 创建详细流程
Feign就是通过扫描添加了@FeignClient注解的接口,然后一步步生成代理对象,具体流程如下:
后续在请求时,通过代理对象的FeignInvocationHandler进行拦截,并根据对应方法进行处理器的分发,完成后续的http请求操作。
4. @FeignClient属性
-
name
定义当前客户端Client的名称。如果项目使用了Ribbon,name属性会作为微服务的名称,用于服务发现。 -
value
等同于name属性。 -
url
配置指定服务的地址。 -
path
配置指定接口的请求路径。 -
configuration
Feign配置类,可以自定义Feign的Encoder、Decoder、LogLevel、Contract等。 -
fallback
定义容错的处理类,当调用远程接口失败或超时时,会调用对应接口的容错逻辑,fallback指定的类必须实现@FeignClient标记的接口。 -
fallbackFactory
工厂类,用于生成fallback类实例,通过这个属性我们可以实现每个接口通用的容错逻辑,减少重复的代码。 -
decode404
当发生http 404错误时,如果该字段为true,会调用decoder进行解码,否则抛出FeignException。
四、Feign使用
1. 常规调用
- 加入Fegin的依赖
<!--fegin组件-->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
- 在启动类需要添加@EnableFeignClients
@SpringBootApplication
@EnableDiscoveryClient
@EnableFeignClients//开启Fegin
public class OrderApplication {}
- 使用添加@FeignClient,定义远程服务的信息
@FeignClient("service-product")//声明调用的提供者的name服务名
public interface ProductService {
//指定调用提供者的哪个方法
//@FeignClient + @GetMapping 就是一个完整的请求路径 http://service-product/product/{pid}
@GetMapping(value = "/product/{pid}")
Product findByPid(@PathVariable("pid") Integer pid);
}
- 服务消费者-订单服务,修改Controller代码,并启动验证
@RestController
@Slf4j
public class OrderController {
@Autowired
private OrderService orderService;
@Autowired
private ProductService productService;
//准备买1件商品
@GetMapping("/order/prod/{pid}")
public Order order(@PathVariable("pid") Integer pid) {
log.info(">>客户下单,这时候要调用商品微服务查询商品信息");
//通过fegin调用商品微服务
Product product = productService.findByPid(pid);
log.info(">>商品信息,查询结果:" + JSON.toJSONString(product));
Order order = new Order();
order.setUid(1);
order.setUsername("测试用户");
order.setPid(product.getPid());
order.setPname(product.getPname());
order.setPprice(product.getPprice());
order.setNumber(1);
orderService.save(order);
return order;
}
}
- 重启order微服务,查看效果
2.日志打印
Feign 提供了日志打印功能,我们可以通过配置来调整日志级别,从而了解 Feign 中 Http 请求的细节。
说白了就是对Feign接口的调用情况进行监控和输出。
配置日志Bean:
import feign.Logger;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class FeignConfig {
@Bean
Logger.Level feignLoggerLevel() {
return Logger.Level.FULL;
}
}
YML文件里需要开启日志的Feign客户端
logging:
level:
# feign日志以什么级别监控哪个接口
com.gzl.cn.service.PaymentFeignService: debug
3. 添加Header
以下提供了四种方式:
1.在@RequestMapping中添加,如下:
@FeignClient(name="custorm",fallback=Hysitx.class)
public interface IRemoteCallService {
@RequestMapping(value="/custorm/getTest",method = RequestMethod.POST,
headers = {"Content-Type=application/json;charset=UTF-8"})
List<String> test(@RequestParam("names") String[] names);
}
2:在方法参数前面添加@RequestHeader注解,如下:
@FeignClient(name="custorm",fallback=Hysitx.class)
public interface IRemoteCallService {
@RequestMapping(value="/custorm/getTest",method = RequestMethod.POST,
headers = {"Content-Type=application/json;charset=UTF-8"})
List<String> test(@RequestParam("names")@RequestHeader("Authorization") String[] names);
}
设置多个属性时,可以使用Map,如下:
@FeignClient(name="custorm",fallback=Hysitx.class)
public interface IRemoteCallService {
@RequestMapping(value="/custorm/getTest",method = RequestMethod.POST,
headers = {"Content-Type=application/json;charset=UTF-8"})
List<String> test(@RequestParam("names") String[] names, @RequestHeader MultiValueMap<String, String> headers);
}
3.使用@Header注解,如下:
@FeignClient(name="custorm",fallback=Hysitx.class)
public interface IRemoteCallService {
@RequestMapping(value="/custorm/getTest",method = RequestMethod.POST)
@Headers({"Content-Type: application/json;charset=UTF-8"})
List<String> test(@RequestParam("names") String[] names);
}
4.实现RequestInterceptor接口(拦截器),如下:
@Configuration
public class FeignRequestInterceptor implements RequestInterceptor {
@Override
public void apply(RequestTemplate temp) {
temp.header(HttpHeaders.AUTHORIZATION, "XXXXX");
}
}