Spring RestTemplate 利用拦截器配置 HTTP请求 Header

Spring RestTemplate经常被用作客户端向Restful API发送各种请求,也许你也碰到过这种需求,很多请求都需要用到相似或者相同的Http Header。如果在每次请求之前都把Header填入HttpEntity/RequestEntity,这样的代码会显得十分冗余。

Spring提供了ClientHttpRequestInterceptor接口,可以对请求进行拦截,并在其被发送至服务端之前修改请求或是增强相应的信息。下面是一个简单的例子:

实现ClientHttpRequestInterceptor接口

// 不是必要的
@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);
  }
}


将自定义拦截器添加到RestTemplate实例

@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等通用的字段。

// 客户端代码
restTemplate.getForObject(SERVER_URL, Example.class);

// 服务端代码
// 如果服务端也是用Spring RestController/MVC 实现,利用@RequestHeader注解,即可获取之前添加的actionId字段了
@RequestMapping(value = "/example")
public Example example(@RequestHeader("actionId") String actionId) {
    //业务逻辑
}

欢迎各路高手留言讨论!
也欢迎关注我的简书



你可能感兴趣的:(技术分享)