@RequestBody和@ResponseBody的使用情形以及RestTemplate的http报文转换
@RequestBody和@ResponseBody两个注解,分别完成请求报文到对象和对象到响应报文的转换。
@RequestBody
1、@requestBody注解常用来处理content-type不是默认的application/x-www-form-urlcoded编码的内容,
比如说:application/json或者是application/xml等。一般情况下来说常用其来处理application/json类型。
作用:
- 注解用于将Controller的方法参数,根据HTTP Request Header的
content-Type
的内容,通过适当的HttpMessageConverter转换为JAVA类 - 再把HttpMessageConverter返回的对象数据绑定到 controller中方法的参数上。
说明:request的body部分的数据编码格式由header部分的Content-Type指定;
2、通过@requestBody可以将请求体中的JSON字符串绑定到相应的bean上,当然,也可以将其分别绑定到对应的字符串上。例如说以下情况:
- $.ajax({
- url:"/login",
- type:"POST",
- data:'{"userName":"admin","pwd","admin123"}',
- content-type:"application/json charset=utf-8",
- success:function(data){
- alert("request success ! ");
- }
- });
- @requestMapping("/login")
- public void login(@requestBody String userName,@requestBody String pwd){
- System.out.println(userName+" :"+pwd);
- }
这种情况是将JSON字符串中的两个变量的值分别赋予了两个字符串,但是呢假如我有一个User类,拥有如下字段:
String userName;
String pwd;
那么上述参数可以改为以下形式:@requestBody User user 这种形式会将JSON字符串中的值赋予user中对应的属性上
需要注意的是,JSON字符串中的key必须对应user中的属性名,否则是请求不过去的。
另外这里要注意其实 @RequestBody接收的是一个Json对象的字符串,而不是一个Json对象。
Json对象和Json字符串的区别就是有没有定义的时候有没有单引号的。。。
如果ajax请求传的是Json对象,后来用 JSON.stringify(data)的方式就能将对象变成字符串。
同时ajax请求的时候也要指定dataType: "json",contentType:"application/json" 这样就可以轻易的将一个对象或者List传到Java端,使用@RequestBody即可绑定对象或者List.。
- <script type="text/javascript">
- $(document).ready(function(){
- var saveDataAry=[];
- var data1={"userName":"test","address":"gz"};
- var data2={"userName":"ququ","address":"gr"};
- saveDataAry.push(data1);
- saveDataAry.push(data2);
- $.ajax({
- type:"POST",
- url:"user/saveUser",
- dataType:"json",
- contentType:"application/json",
- data:JSON.stringify(saveData),
- success:function(data){
- }
- });
- });
- </script>
@ResponseBody
该注解用于将Controller的方法返回的对象,根据HTTP Request Header的Accept
的内容,
通过适当的HttpMessageConverter
转换为指定格式后,写入到Response对象的body数据区。
通常是在使用 @RequestMapping 后,返回值通常解析为跳转路径,加上 @Responsebody 后返回结果不会被解析为跳转路径,而是直接写入HTTP 响应正文中。
HTTP 请求和响应是基于文本的,意味着浏览器和服务器通过交换原始文本进行通信。但是,使用 Spring,controller 类中的方法返回纯 ‘String’ 类型和域模型(或其他 Java 内建对象)。
如何将对象序列化/反序列化为原始文本?这由HttpMessageConverter 处理。
Http请求和响应报文本质上都是一串字符串,当请求报文来到java世界,它会被封装成为一个ServletInputStream的输入流,供我们读取报文。响应报文则是通过一个ServletOutputStream的输出流,来输出响应报文。
我们从流中,只能读取到原始的字符串报文,同样,我们往输出流中,也只能写原始的字符。
而在java世界中,处理业务逻辑,都是以一个个有业务意义的对象为处理维度的,那么在报文到达SpringMVC和从SpringMVC出去,都存在一个字符串到java对象的阻抗问题。
这一过程,不可能由开发者手工转换。我们知道,在Struts2中,采用了OGNL来应对这个问题,而在SpringMVC中,它是HttpMessageConverter机制。
请看:
SpringMVC源码剖析(五)-消息转换器HttpMessageConverter
HttpMessageConverter(消息转换器 )和@responsebody使用
RestTemplate中http报文转换处理
在RestTemplate中,
我们知道,调用reseful接口传递的数据内容是json格式的字符串,返回的响应也是json格式的字符串。
然而restTemplate.postForObject
方法的请求参数RequestBean
和返回参数ResponseBean
却都是java类。是RestTemplate
通过HttpMessageConverter
自动帮我们做了转换的操作。
默认情况下RestTemplate
自动帮我们注册了一组HttpMessageConverter
用来处理一些不同的contentType
的请求。
如StringHttpMessageConverter
来处理text/plain
;MappingJackson2HttpMessageConverter
来处理application/json
;MappingJackson2XmlHttpMessageConverter
来处理application/xml
。
你可以在org.springframework.http.converter
包下找到所有spring帮我们实现好的转换器。
如果现有的转换器不能满足你的需求,你还可以自己实现。
请看:如何使用RestTemplate访问restful服务
注意:
StringHttpMessageConverter
默认使用的字符集是ISO-8859-1
,在遇到中文的时候会有乱码,所以需要移除RestTemplate默认的StringHttpMessageConverter
修改字符字符集后重新设置。
spring的json转换器默认使用的是Jackson,json字符串和对应的Entity如果有字段对不上就会报错,这个有点不符合国情,而FastJson则不会报错,所以很多时候都会用FastJSON替换默认的Jackson。
配置示例:
- @Configuration
- public class RestAutoConfig {
- public static class RestTemplateConfig {
- @Bean//负载均衡的restTemplate
- @LoadBalanced //spring 对restTemplate bean进行定制,加入loadbalance拦截器进行ip:port的替换
- //"http://user/getusername,就能解析成http://127.0.0.1:8083//getusername
- RestTemplate lbRestTemplate(HttpClient httpclient) {
- RestTemplate template = new RestTemplate(new HttpComponentsClientHttpRequestFactory(httpclient));
- template.getMessageConverters().add(,new StringHttpMessageConverter(Charset.forName("utf-8")));
- template.getMessageConverters().add(,new FastJsonHttpMessageConvert5());
- return template;
- }
- @Bean //直连的restTemplat,这时只能使用http://127.0.0.1:8083//getusername地址,不能解析http://user/getusername
- RestTemplate directRestTemplate(HttpClient httpclient) {
- RestTemplate template = new RestTemplate(new HttpComponentsClientHttpRequestFactory(httpclient));
- template.getMessageConverters().add(,new StringHttpMessageConverter(Charset.forName("utf-8")));
- template.getMessageConverters().add(,new FastJsonHttpMessageConvert5());
- return template;
- }
- // FastJsonHttpMessageConvert4有一个bug,它是默认支持MediaType.ALL,spring在处理MediaType.ALL的时候会识别成字节流,而不是json,这里就对他进行改造和处理
- public static class FastJsonHttpMessageConvert5 extends FastJsonHttpMessageConverter4{
- static final Charset DEFAULT_CHARSET = Charset.forName("UTF-8");
- public FastJsonHttpMessageConvert5(){
- setDefaultCharset(DEFAULT_CHARSET);
- setSupportedMediaTypes(Arrays.asList(MediaType.APPLICATION_JSON,new MediaType("application","*+json")));
- }
- }
- }
- }
HTTPclient Bean获取类:
- @Configuration
- @ConditionalOnClass({HttpClient.class})
- @EnableConfigurationProperties(HttpClientProperties.class)
- public class HttpClientAutoConfiguration {
- private final HttpClientProperties properties;
- public HttpClientAutoConfiguration(HttpClientProperties properties){
- this.properties = properties;
- }
- /**
- * httpclient bean 的定义
- * @return
- */
- @Bean
- @ConditionalOnMissingBean(HttpClient.class)
- public HttpClient httpClient() {
- RequestConfig requestConfig = RequestConfig.custom()
- .setConnectTimeout(properties.getConnectTimeOut())
- .setSocketTimeout(properties.getSocketTimeOut()).build();// 构建requestConfig
- HttpClient client = HttpClientBuilder.create().setDefaultRequestConfig(requestConfig)
- .setUserAgent(properties.getAgent())
- .setMaxConnPerRoute(properties.getMaxConnPerRoute())
- .setMaxConnTotal(properties.getMaxConnTotaol())
- .build();
- return client;
- }
- }
HTTPClient参数类:
- @ConfigurationProperties(prefix="spring.httpclient")
- public class HttpClientProperties {
- private Integer connectTimeOut = ;
- private Integer socketTimeOut = ;
- private String agent = "agent";
- private Integer maxConnPerRoute = ;
- private Integer maxConnTotaol = ;
- public Integer getConnectTimeOut() {
- return connectTimeOut;
- }
- public void setConnectTimeOut(Integer connectTimeOut) {
- this.connectTimeOut = connectTimeOut;
- }
- public Integer getSocketTimeOut() {
- return socketTimeOut;
- }
- public void setSocketTimeOut(Integer socketTimeOut) {
- this.socketTimeOut = socketTimeOut;
- }
- public String getAgent() {
- return agent;
- }
- public void setAgent(String agent) {
- this.agent = agent;
- }
- public Integer getMaxConnPerRoute() {
- return maxConnPerRoute;
- }
- public void setMaxConnPerRoute(Integer maxConnPerRoute) {
- this.maxConnPerRoute = maxConnPerRoute;
- }
- public Integer getMaxConnTotaol() {
- return maxConnTotaol;
- }
- public void setMaxConnTotaol(Integer maxConnTotaol) {
- this.maxConnTotaol = maxConnTotaol;
- }
- }
@RequestBody和@ResponseBody的使用情形以及RestTemplate的http报文转换的更多相关文章
- springmvc实现json交互 -requestBody和responseBody
json数据交互 1.为什么要进行json数据交互 json数据格式在接口调用中.html页面中较常用,json格式比较简单,解析还比较方便. 比如:webservice接口,传输json数据. 2. ...
- @RequestBody 和@ResponseBody 注解详解
简介: @RequestBody 作用: i) 该注解用于读取Request请求的body部分数据,使用系统默认配置的HttpMessageConverter进行解析,然后把相应的数据绑定到要返回的对 ...
- 在SpringMVC中使用@RequestBody和@ResponseBody注解处理json时,报出HTTP Status 415的解决方案
我在使用SpringMVC的@RequestBody和@ResponseBody注解处理JSON数据的时候,总是出现415的错误,说是不支持所提交数据格式,我在页面中使用了JQuery的AJAX来发出 ...
- Spring MVC源码(三) ----- @RequestBody和@ResponseBody原理解析
概述 在SpringMVC的使用时,往往会用到@RequestBody和@ResponseBody两个注解,尤其是处理ajax请求必然要使用@ResponseBody注解.这两个注解对应着Contro ...
- @RequestBody、@ResponseBody注解是如何将输入输出转换成json的
@RequestBody.@ResponseBody注解,可以直接将输入解析成Json.将输出解析成Json,但HTTP 请求和响应是基于文本的,意味着浏览器和服务器通过交换原始文本进行通信,而这里其 ...
- @RequestParam,@RequestBody,@ResponseBody,@PathVariable注解的一点小总结
一.前提知识: http协议规定一次请求对应一次响应,根据不同的请求方式,请求的内容会有所不同: 发送GET请求是没有请求体的,参数会直接拼接保留到url后一并发送: 而POST请求是带有请求体的,带 ...
- SpringMVC使用@PathVariable,@RequestBody,@ResponseBody,@RequestParam,@InitBinder
@Pathvariable public ResponseEntity<String> ordersBack( @PathVariable String reqKey, ...
- @RequestBody和@ResponseBody
@RequestBody 将HTTP请求正文转换为适合的HttpMessageConverter对象. @ResponseBody 将内容或对象作为 HTTP 响应正文返回,并调用适合HttpMess ...
- SpringMVC中注解@RequestBody和@ResponseBody的使用区别
首先上源码 在面试时经常会问到我们如何使用SpringMVC将Http请求转换为java对象,或者又是问如何将结果转换为java的呢? SpringMVC在接收到请求之后HandlerMapping像 ...
随机推荐
- ArrayList和Vector的区别?
ArrayList和Vector的区别? 解答:同步性:Vector是线程安全的,也就是说是同步的,而ArrayList是线程不安全的,不是同步的:数据增长:当需要增长时,Vector默认增长为原来一 ...
- bootstrap基础学习九篇
现在学学bootstrap响应式实用工具 Bootstrap 提供了一些辅助类,以便更快地实现对移动设备友好的开发.这些可以通过媒体查询结合大型.小型和中型设备,实现内容对设备的显示和隐藏. 需要谨慎 ...
- 你 get 了无数技能,为什么一事无成
前 几日看到阮一峰老师的发的一句话,颇有感慨,「你只是坐在电脑前,往网上发表了一段文字或者一张图片,随便什么,就能够接触到多少陌生的灵魂.这就是我热 爱互联网的原因」.我打心底认为这是一个最好的时代, ...
- PrintArea打印,@media screen解决移动web开发的多分辨率问题,@media print设置打印的样式
PrintArea打印,局部DIV打印插件,依赖JQuery. github:https://github.com/RitsC/PrintArea 当打印时需要临时改变页面布局,可以使用 @media ...
- 用12个例子全面示范Angular的模板语法
template分支,用12个例子全面示范Angular的模板语法 // 使用方法 git clone https://git.oschina.net/mumu-osc/learn-component ...
- 使用ghost硬盘对拷备份系统
公司有台server装了OA系统.要备份数据.同一时候假设系统出错之后可以及时回复.所以有买了块同型号硬盘. 用ghost的硬盘对拷功能,将原硬盘的系统和数据拷到新硬盘上.新硬盘挂到server上.当 ...
- 74、shape 画圆 加 边框
<?xml version="1.0" encoding="utf-8"?> <!--<shape xmlns:android=&quo ...
- Jmeter--CSV Data Set Config 参数化配置
博客首页:http://www.cnblogs.com/fqfanqi/ 设置界面如下: Filename:参数文件名,一般是.csv和.txt文件.绝对路径和相对路径都可以,为了便于脚本迁移,建议使 ...
- Python2 显示 unicode
用户想要看的是 u'中文' 而不是 u'\u4e2d\u6587',但是在 Python2 中有时并不能实现. 转译 转义字符是这样一个字符,标志着在一个字符序列中出现在它之后的后续几个字符采取一种替 ...
- 请写出用于校验HTML文本框中输入的内容全部为数字的javascript代码
<%@ page contentType="text/html;charset=UTF-8" language="java" %> <html ...