Spring RestTemplate常常被用做客户端向Restful API发送各类请求,也许你也碰到过这种需求,不少请求都须要用到类似或者相同的Http Header。若是在每次请求以前都把Header填入HttpEntity/RequestEntity,这样的代码会显得十分冗余。app
Spring提供了ClientHttpRequestInterceptor接口,能够对请求进行拦截,并在其被发送至服务端以前修改请求或是加强相应的信息。下面是一个简单的例子:ide
// 不是必要的 @Component public class ActionTrackInterceptor implements ClientHttpRequestInterceptor { @Autowired ActionIdGenerator actionIdGenerator; @Override public ClientHttpResponse intercept(HttpRequest request, byte[] body, ClientHttpRequestExecution execution) throws IOException { HttpHeaders headers = request.getHeaders(); // 加入自定义字段 headers.add("actionId", actionIdGenerator.generate()); // 保证请求继续被执行 return execution.execute(request, body); } }
@Configuration public class ClientConfig { // 注入拦截器。拦截器也能够不声明为Bean, 直接在这里新建实例 @Autowired ActionTrackInterceptor actionTrackInterceptor; // 声明为Bean,方便应用内使用同一实例 @Bean public RestTemplate restTemplate() { RestTemplate restTemplate = new RestTemplate(); // 把自定义的ClientHttpRequestInterceptor添加到RestTemplate,可添加多个 restTemplate.setInterceptors(Collections.singletonList(actionTrackInterceptor)); return restTemplate; } }
前期的工做已经完成了,如今使用这个RestTemplate实例发送请求,就会在Header中带上“actionId”这个字段了,固然你能够配置更多的诸如Accept, Content-Type等通用的字段。spa
// 客户端代码 restTemplate.getForObject(SERVER_URL, Example.class); // 服务端代码 // 若是服务端也是用Spring RestController/MVC 实现,利用@RequestHeader注解,便可获取以前添加的actionId字段了 @RequestMapping(value = "/example") public Example example(@RequestHeader("actionId") String actionId) { //业务逻辑 }