RestTemplate是spring内置的http请求封装,它提供一套接口,实际用Java常用http连接库来实现这套接口,java常用的实现http请求实现库有以下三种:
在使用spring的情况下,http请求直接使用RestTemplate是不错的选择。
使用RestTemplate发起http请求的时候,Rest服务提供者没有什么特殊要求,直接按照传统的SpringMVC的Controller层实现方式实现即可。
举例:
@RestController
@RequestMapping("/user")
public class UserController {
private Logger logger = LoggerFactory.getLogger(UserController.class);
@PostMapping("/add")
public UserBean add(UserBean userBean) {
logger.info("request param:{}", JSON.toJSON(userBean));
return userBean;
}
@GetMapping("/{id}")
public UserBean getUser(@PathVariable("id") Long userId, @RequestHeader("token") String token) {
logger.info("get user id={},token={}", userId, token);
UserBean userBean = new UserBean();
userBean.setAge(33);
userBean.setId(userId);
userBean.setName("张山");
return userBean;
}
}
使用RestTemplate前先得做一些初始化处理,比如指定http客户端工厂类、设置超时时间、响应参数转换器等。
@Configuration
@ConditionalOnClass(value = {RestTemplate.class, HttpClient.class})
public class RestTemplateConfiguration {
@Value("${remote.maxTotalConnect:0}")
private int maxTotalConnect; //连接池的最大连接数默认为0
@Value("${remote.maxConnectPerRoute:200}")
private int maxConnectPerRoute; //单个主机的最大连接数
@Value("${remote.connectTimeout:2000}")
private int connectTimeout; //连接超时默认2s
@Value("${remote.readTimeout:30000}")
private int readTimeout; //读取超时默认30s
//创建HTTP客户端工厂
private ClientHttpRequestFactory createFactory() {
if (this.maxTotalConnect <= 0) {
SimpleClientHttpRequestFactory factory = new SimpleClientHttpRequestFactory();
factory.setConnectTimeout(this.connectTimeout);
factory.setReadTimeout(this.readTimeout);
return factory;
}
HttpClient httpClient = HttpClientBuilder.create().setMaxConnTotal(this.maxTotalConnect)
.setMaxConnPerRoute(this.maxConnectPerRoute).build();
HttpComponentsClientHttpRequestFactory factory = new HttpComponentsClientHttpRequestFactory(
httpClient);
factory.setConnectTimeout(this.connectTimeout);
factory.setReadTimeout(this.readTimeout);
return factory;
}
//初始化RestTemplate,并加入spring的Bean工厂,由spring统一管理
@Bean
@ConditionalOnMissingBean(RestTemplate.class)
public RestTemplate getRestTemplate() {
RestTemplate restTemplate = new RestTemplate(this.createFactory());
List<HttpMessageConverter<?>> converterList = restTemplate.getMessageConverters();
//重新设置StringHttpMessageConverter字符集为UTF-8,解决中文乱码问题
HttpMessageConverter<?> converterTarget = null;
for (HttpMessageConverter<?> item : converterList) {
if (StringHttpMessageConverter.class == item.getClass()) {
converterTarget = item;
break;
}
}
if (null != converterTarget) {
converterList.remove(converterTarget);
}
converterList.add(1, new StringHttpMessageConverter(StandardCharsets.UTF_8));
//加入FastJson转换器
converterList.add(new FastJsonHttpMessageConverter4());
return restTemplate;
}
}
StringHttpMessageConverter
默认使用的字符集是ISO-8859-1
,在遇到中文的时候会有乱码,所以需要移除RestTemplate默认的StringHttpMessageConverter
修改字符字符集后重新设置。
spring的json转换器默认使用的是Jackson,json字符串和对应的Entity如果有字段对不上就会报错,这个有点不符合国情,而FastJson则不会报错,所以很多时候都会用FastJSON替换默认的Jackson。
@Configuration
public class FastjsonConfiguration {
@Bean
public HttpMessageConverters fastjsonConverter() {
FastJsonConfig fastJsonConfig = new FastJsonConfig();
//自定义格式化输出
fastJsonConfig.setSerializerFeatures(SerializerFeature.PrettyFormat,
SerializerFeature.WriteNullStringAsEmpty, SerializerFeature.WriteNullNumberAsZero);
FastJsonHttpMessageConverter4 fastjson = new FastJsonHttpMessageConverter4();
fastjson.setFastJsonConfig(fastJsonConfig);
return new HttpMessageConverters(fastjson);
}
}
MultiValueMap
传参 @Test
public void multiValueMapParam() {
MultiValueMap<String, String> requestParam = new LinkedMultiValueMap<>();
requestParam.set("name", "张三");
UserBean result = restTemplate
.postForObject("http://127.0.0.1:8280/user/add", requestParam, UserBean.class);
logger.info("result:{}", JSON.toJSON(result));
}
Entity
传参 @Test
public void restClient() {
UserBean userBean = new UserBean();
userBean.setName("王五");
String result = restTemplate
.postForObject("http://127.0.0.1:8280/user/add", userBean, String.class);
logger.info("result:{}", result);
}
以Entity传参,接收端Controller以Entity接收参数的时候需加上@RequestBody
注解,否则接收不到参数。
@PostMapping("/add")
public UserBean add(@RequestBody UserBean userBean) {
logger.info("request param:{}", JSON.toJSON(userBean));
return userBean;
}
/** * 普通 GET请求,请求参数在路径中 */
@Test
public void getUser() {
long userId = 32L;
UserBean result = restTemplate.getForObject(
"http://127.0.0.1:8280/user/{id}",
UserBean.class,
userId);
logger.info("result={}", JSON.toJSONString(result));
}
实际项目中远程接口调用很多时候是需要授权访问的,授权信息通常设置在http请求头里面,RestTemplate封装的get请求方法里面没有设置header的地方,那么我们是否对这种需要有header的get请求就不能使用RestTemplate了呢,当然不是。
那几个get请求封装方法重载程度较高,要往请求里面加请求头不容易,那我们就用里面的exchange方法,这个方法的封装程度是不高的,里面直接设置请求方式,并可以设置请求头。
/** * 带header的GET请求 */
@Test
public void getHasHeader() {
long userId = 32L;
HttpHeaders headers = new HttpHeaders();
headers.add("token", "123");
ResponseEntity<UserBean> response = restTemplate.exchange(
"http://127.0.0.1:8280/user/{id}",
HttpMethod.GET,
new HttpEntity<String>(headers),
UserBean.class,
userId);
logger.info("response={}", JSON.toJSONString(response.getBody()));
}
原文地址:http://www.itclj.com/blog/5925894681c06e672f942ad6
demo地址:https://github.com/clj198606061111/spring-boot-rest-template-demo