前言
在Spring Cloud Netflix棧中,各個微服務都是以HTTP接口的形式暴露自身服務的,因此在調用遠程服務時就必須使用HTTP客戶端。我們可以使用JDK原生的URLConnection、Apache的Http Client、Netty的異步HTTP Client, Spring的RestTemplate。公司目前使用的是Feign。下面來和大家一起學習下feign調用中兩種Header傳參方式。
在請求攔截器中統一配置
每次Feign調用中,需要傳遞一些安全校驗參數,比如說token,bizId啥的,如果單獨在每次調用的時候去配置這些重複的邏輯,顯然不合適。所以我們可以在Feign的攔截器中統一配置這些權限參數。如何配置呢?其實很簡單。
import org.springframework.context.annotation.Configuration;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import feign.RequestInterceptor;
import feign.RequestTemplate;
import javax.servlet.http.HttpServletRequest;
import java.util.Enumeration;
@Configuration
public class FeignConfiguration implements RequestInterceptor {
@Override
public void apply(RequestTemplate template) {
ServletRequestAttributes attributes = (ServletRequestAttributes)
RequestContextHolder.getRequestAttributes();
HttpServletRequest request = attributes.getRequest();
Enumeration<String> headerNames = request.getHeaderNames();
if (headerNames != null) {
while (headerNames.hasMoreElements()) {
String name = headerNames.nextElement();
String values = request.getHeader(name);
template.header(name, values);
}
}
}
}
然後再feignClient中配置自定義的配置類,覆蓋默認的配置
@Component
@FeignClient(value = "feignTest" ,configuration = FeignConfiguration.class)
public interface FeignTestService {
}
這樣就所有的Feign調用都會在Header調用中加上request中傳遞過來的參數。不需要單獨在每次Feign調用重複添加參數。
通過@RequestHeader註解
上面也說到,我們在攔截器中獲取request傳遞過來的參數,然後再放到header中。在攔截器中配置的一般都是公用的一些參數。但是如果一個服務接口,需要一些特殊參數。但是其他服務不需要這個參數,這時候我們不必在攔截器中配置其他服務不需要的參數,只需在需要的服務接口上加上@RequestHeader註解,然後在feign調用的時候把參數傳過來就ok了。
@PostMapping(value = "/orderDetails")
public ReturnInfo getOrderDetail(@RequestHeader(name = "id") String orderId);
比如說之前在quartz中通過Feign調用其他服務獲取優惠券信息,項目中封裝的方法獲取的attributes 是空的。
ServletRequestAttributes attributes = (ServletRequestAttributes)
RequestContextHolder.getRequestAttributes();
這時候就不用攔截器了,直接在方法入參上加上 @RequestHeader註解。
總結
這裏簡單介紹下Feign調用的傳參,實際在項目中,大家對Feign調用方法都有自己的封裝,代碼和上面的可能不一樣,但是原理都是差不多的。