Java Feign
Feign其实也是通过HTTP请求来实现的通信。那么自然绕不开HTTP相关的东西,比如很多系统中权限校验都是通过Header中的参数来实现,需要将前端传过来的header转发到目标服务,这里记录一下关于Header的设置。
下面提到的同步/异步 只是记录一下遇到问题的情景, 异步方法的实现方式同样适用于同步方法

同步

同步方法一: 通过拦截器

这种方法是比较通用而且也比较常见的,通过实现Feign的RequestInterceptor接口,重写里面的apply方法,为RequestTemplate添加请求头信息
代码如下

  1. @Component
  2. public class FeignClientsConfigurationCustom implements RequestInterceptor {
  3. @Override
  4. public void apply(RequestTemplate requestTemplate) {
  5. // 此种方式是线程安全的
  6. ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder
  7. .getRequestAttributes();
  8. // 不为空时取出请求中的header 原封不动的设置到feign请求中
  9. if (null != attributes) {
  10. HttpServletRequest request = attributes.getRequest();
  11. if (null != request) {
  12. // 遍历设置 也可从request取出整个Header 写到RequestTemplate 中
  13. Enumeration<String> headerNames = request.getHeaderNames();
  14. if (headerNames != null) {
  15. while (headerNames.hasMoreElements()) {
  16. String name = headerNames.nextElement();
  17. String values = request.getHeader(name);
  18. requestTemplate.header(name, values);
  19. }
  20. }
  21. }
  22. }
  23. }
  24. }

完成上述代码之后 在FeignClient注解中加入 configuration = FeignClientsConfigurationCustom.class 即可

  1. @FeignClient(name = "testClient", configuration = FeignClientsConfigurationCustom.class)
  2. public interface testServer{
  3. }

同步方法二:@Headers注解

这种方法比较适用于一些不变的参数,如Content-Type等

  1. @FeignClient(name = "testClient", configuration = FeignClientsConfigurationCustom.class)
  2. public interface testServer{
  3. @GetMapping("/test")
  4. @Headers({"Content-Type: application/json","Accept: application/json"})
  5. String test(@RequestParam String param);
  6. }

异步

此处说的异步场景是通过Spring中的@Async实现的

  1. // 带有@Async注解,异步调用Feign
  2. @Autowired
  3. private TestService testService;
  4. @GetMapping("/test")
  5. public String test(){
  6. // 此方法是一个异步方法, 在该方法中调用了Feign服务
  7. testService.testMethod();
  8. return "Hello world!";
  9. }

此时的业务场景是 前端发送一个服务,异步地调用其他微服务的方法, 由于此方法执行耗时可能会比较长,而且对用户来说没有下一步操作,所以直接return掉,那么主线程因return而关闭,此时在刚刚的FeignConfig中就无法获取到请求头了。
Feign 请求头设置与传递问题 - 图1
如图所示,红色方框圈起来的地方都不为null,但是最终获取到的Header是一个空的Map。
所以猜测是因为主线程退出触发了JVM的回收机制。
那么此时的情况将是 主线程已经退出,子线程没有执行完
所以显然,此时不能通过这种方式传递Header了。

异步方法一:线程私有变量ThreadLocal。

既然无法直接通过获取HttpServletRequest来获取Header,那么可以稍微改造一下,在原来的基础上添加一个拦截器。
所有的请求过来的时候,在拦截器中将Header先取出来,然后设置到本线程私有的Map中。
原来的apply方法在提交请求的时候再通过ThreadLocal提供的remove方法,清除掉。
只要把对该Map的操作封装一个工具类,工具类中实现get/set方法即可。
其实这种方式就是换了一个地方保存请求头,因此实用性与便捷性都还可以。

异步方法二:通过传参

该方式是在方法执行前,先将需要的参数取出来,比如需要一个token 就在Header中取出token,需要一个Content-Type就取出Content-Type。
然后将取出来的值作为参数传递到待执行的方法中。
该方法所调用的Feign接口需要做一个改造,在参数中添加带有@RequestHeader的注解,该注解表示将变量放在请求头,而不是请求的参数或者请求体里面。

  1. @FeignClient(name = "testClient", configuration = FeignClientsConfigurationCustom.class)
  2. public interface testServer{
  3. @GetMapping("/test")
  4. String test(@RequestParam String param,@RequestHeader String token);
  5. }

这种方法对于原来代码的改动较小,如果异步的场景比较少的话可以选择这种方法。
但是如果项目中用到了较多的异步方法,那么就需要异步方法一里面的拦截器 + ThreadLocal + RequestInterceptor了。