1.引入httpclient依赖,在pom.xml加入
org.apache.httpcomponents
httpclient
4.3.3
2.在MvcConfig注册RestTemplate Bean.
@Bean
@Scope(value = WebApplicationContext.SCOPE_SESSION,proxyMode = ScopedProxyMode.INTERFACES)
public RestTemplate restTemplate(){
ClientHttpRequestFactory clientHttpRequestFactory = new HttpComponentsClientHttpRequestFactory();
return new RestTemplate(clientHttpRequestFactory);
}
因为这样的模板工具是属于Web层的东西,放在Web配置更合理.再有,对于scope为session的Bean也必须在WebApplicationContext下使用,如果是在常规的spring容器(即非web环境)下使用会抛IllegalStateException,这点spring-framework-reference有提及.至于这里为什么要使用session的Bean,下面会做分析.
@Controller
public class DefaultController {
@Autowired
private RestOperations restOperations;
@RequestMapping("/test")
public String test(String url){
//get请求
String result=restOperations.getForObject(url, String.class);
System.out.println(result);
//post请求,注意:参数不是传给uriVariables,uriVariables是用于设置uri参数变量的
final MultiValueMap params = new LinkedMultiValueMap<>();
params.add("username", "124124");
params.add("age", "28");
//params.add("file", new MockMultipartFile("test.txt", "sfasfsf".getBytes("UTF-8")));
result = restOperations.postForObject(url, params, String.class);
//result = restOperations.postForObject(url, new HttpEntity<>(params, null), String.class);
return "user/add";
}
}
这里注入RestTemplate的代理实例(因为上面代理是通过JDK生成的,所以这里要用接口注入)供Controller的方法使用.这里的意图就是,传入一个url,然后根据这个url发起请求(不传入参数,是为了简单),返回字符串形式的响应. 对于普通的请求参数(不含上传文件),应使用MultiValueMap类型,理由如下,先看org.springframework.http.converter.FormHttpMessageConverter#write
public void write(MultiValueMap map, MediaType contentType, HttpOutputMessage outputMessage)
throws IOException, HttpMessageNotWritableException {
if (!isMultipart(map, contentType)) {
writeForm((MultiValueMap) map, contentType, outputMessage);
} else {
writeMultipart((MultiValueMap) map, outputMessage);
}
}
private boolean isMultipart(MultiValueMap map, MediaType contentType) {
if (contentType != null) {
return MediaType.MULTIPART_FORM_DATA.includes(contentType);
}
for (String name : map.keySet()) {
for (Object value : map.get(name)) {
if (value != null && !(value instanceof String)) {
return true;
}
}
}
return false;
}
从isMultipart方法可以看出value不为空且全是String类型才认为不是Multipart请求,参数会以key1=value1&key2=value2形式写入请求体,在springMvc的controller方法public String list(String username,Integer age){}是可以正常映射. 对于Multipart请求,请求的另一种形式 --9EGuVcL0OtjH40Pnd-MjN5Cs6xyUM9FPzX Content-Disposition: form-data; name="username" Content-Type: text/plain;charset=ISO-8859-1 Content-Length: 9 124124 --9EGuVcL0OtjH40Pnd-MjN5Cs6xyUM9FPzX Content-Disposition: form-data; name="age" Content-Type: application/json;charset=UTF-8 28 这种传Multipart形式请求(如果有上传文件,不能用get请求传输,并且要指定为这种Multipart,我也郁闷没用restTemplate成功上传过文件),在springMvc不配MultipartResolver,参数是不能正常解析的.虽然可能不会报错.但只要在servletApplicationContext声明以下Bean就可以
commons-fileupload
commons-fileupload
1.3.1
@Bean
public MultipartResolver multipartResolver() {
CommonsMultipartResolver bean = new CommonsMultipartResolver();
bean.setDefaultEncoding("UTF-8");
return bean;
}
有了MultipartResolver之后,上面的params.add("age","28");改为params.add("age",28);完全没问题
4.测试.就拿前文作为服务端测试例子,此例对于要从/user/list获取数据,用户必须有"user_list"的权限,如果用户没登录,没有这权限,就会返回403,禁止访问. a.先模拟用户登录,在浏览器输入:http://localhost:8081/web2/test?url=http://localhost:8080/web1/user/login?id=1 b.然后在用户登录的前提下去获取数据,再输入:http://localhost:8081/web2/test?url=http://localhost:8080/web1/user/list 这测试都很顺利正常,下面分析为什么要用session的Bean(建议看spring-framework-reference的Session scope): 因为对于不同的用户有不同的会话,如果使用singleton的话,那么会造成所有的用户都共用这个RestTemplate,并不希望有一个用户登录了,另一个用户根本没有登录,就可以前一个用户的身份去发请求,再有新登录的用户也会覆盖旧用户登录的信息. 使用了session的restTemplate Bean即可解决此问题.这样使用,它的有效作用范围就是HTTP会话级别,即不同的会话,DefaultController的restOperations会注入不同的代理.(为什么要使用代理,建议去看spring-framework-reference的Scoped beans as dependencies,代理有两种,这里使用基于JDK).