Spring MVC REST内容协商(同一资源,多种展现方式,html、xml、json等,通过ContentNegotiatingViewResolver)

转自:http://cnge06.iteye.com/blog/1755856

突然对ruby on rails来了兴趣,便捣弄了两天,用脚手架做了个hello world,实质上也就是一行命令的事,的确够效率。看一下Controller,其REST风格实在太简洁了。

 

Ruby代码     收藏代码
  1. # GET /posts  
  2. # GET /posts.json  
  3. def index  
  4.   @posts = Post.all  
  5.   
  6.   respond_to do |format|  
  7.     format.html # index.html.erb  
  8.     format.json { render json: @posts }  
  9.   end  
  10. end  

 一直在关注着spring,如今一般的web项目,从前到后几乎可以用spring直接搞定了。因此,对spring mvc及REST相关内容自然也有所了解。

 

 

于是今天就在想,spring mvc应该也可以对同一资源,实现多种展现才是,前提当然是在同一个方法里。(之前是为json的处理单独写一个Controller的method的,倒也没什么问题。)

就如以下:

 

Java代码     收藏代码
  1. @RequestMapping(value={"/user/{id}","/user/{id}.json"})  
  2.     public ModelAndView show(@PathVariable(value="id")String id){  
  3.         User user=userService.get(Long.valueOf(id));  
  4.         ModelAndView mav=new ModelAndView("show");  
  5.         mav.addObject(user);  
  6.         return mav;  
  7.     }  

 

 当我请求/user/1时,便会回应show的视图,当我请求/user/{id}.json时,便会回应json格式的数据。

 

而spring mvc中ContentNegotiatingViewResolver便能满足这个需求,除了json,还有xml/rss。

 

相关资讯可查看这里(里面还对RESTful中同一资源多种表述进行描述的三种方式进行了讨论。)。

我当前使用spring版本是3.2,在这个版本里,对ContentNegotiatingViewResolver作了修改,原样参照以上链接配置,出现java.lang.ClassCastException: java.lang.String cannot be cast to org.springframework.http.MediaType 的异常。版本原因造成这个问题。

在springsource的论坛里有人询问和解答,请看这里或者这里

 

起作用的配置,例:

 

Java代码     收藏代码
  1. <!-- REST -->  
  2.     <bean class="org.springframework.web.servlet.view.ContentNegotiatingViewResolver">  
  3.         <property name="order" value="1" />  
  4.         <property name="contentNegotiationManager">  
  5.             <bean class="org.springframework.web.accept.ContentNegotiationManager">  
  6.                 <constructor-arg>  
  7.                     <bean class="org.springframework.web.accept.PathExtensionContentNegotiationStrategy">  
  8.                         <constructor-arg>  
  9.                             <map>  
  10.                                 <entry key="json" value="application/json"/>  
  11.                                 <entry key="xml" value="application/xml"/>  
  12.                             </map>  
  13.                         </constructor-arg>  
  14.                     </bean>  
  15.                 </constructor-arg>  
  16.             </bean>  
  17.         </property>  
  18.   
  19.   
  20.         <property name="defaultViews">  
  21.             <list>  
  22.                 <!-- JSON View -->  
  23.                 <bean class="org.springframework.web.servlet.view.json.MappingJacksonJsonView" />  
  24.   
  25.                 <!-- XML View -->  
  26.                 <bean class="org.springframework.web.servlet.view.xml.MarshallingView">  
  27.                     <constructor-arg>  
  28.                         <bean class="org.springframework.oxm.jaxb.Jaxb2Marshaller">  
  29.                             <property name="packagesToScan">  
  30.                                 <list>  
  31.                                     <value>documentLoader.domain</value>  
  32.                                 </list>  
  33.                             </property>  
  34.                         </bean>  
  35.                     </constructor-arg>  
  36.                 </bean>  
  37.             </list>  
  38.         </property>  
  39.     </bean>  
  40.   
  41.     <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">  
  42.         <property name="viewClass" value="org.springframework.web.servlet.view.JstlView" />  
  43.         <property name="prefix" value="/WEB-INF/views/" />  
  44.         <property name="suffix" value=".jsp" />  
  45.     </bean>  

转自:http://www.blogjava.net/badqiu/archive/2009/12/21/306793.html

一.REST内容协商介绍 

RESTful服务中很重要的一个特性即是同一资源,多种表述.也即如下面描述的三种方式:

1.使用http request header: Accept

GET /user/123 HTTP/1.1
Accept: application/xml                 //将返回xml格式数据

GET /user/123 HTTP/1.1
Accept: application/json               //将返回json格式数据

2.使用扩展名

/user/123.xml  将返回xml格式数据
/user/123.json 将返回json格式数据
/user/123.html 将返回html格式数据

3.使用参数

/user/123?format=xml          //将返回xml数据
/user/123?format=json          //将返回json数据


而以上三种各有优缺点:
1.使用Accept header:
   这一种为教科书中通常描述的一种,理想中这种方式也是最好的,但如果你的资源要给用户直接通过浏览器访问(即html展现),那么由于浏览器的差异,发送上来的Accept Header头将是不一样的. 将导致服务器不知要返回什么格式的数据给你. 下面是浏览器的Accept Header

chrome:   
Accept:application/xml,application/xhtml+xml,textml;q=0.9,text/plain;q=0.8,image/png,*/*;q=0.5   
  
firefox:   
Accept:text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8   
  
IE8:   
Accept:image/gif, image/jpeg, image/pjpeg, image/pjpeg, application/x-shockwave-flash, application/x-silverlight, application/x-ms-application, application/x-ms-xbap, application/vnd.ms-xpsdocument, application/xaml+xml, */* 



 

2.使用扩展名
  丧失了同一url多种展现的方式,但现在这种在实际环境中是使用最多的.因为更加符合程序员的审美观.

3.使用参数
  可能由于要编写的字符较多,所以较少使用.

带着上面的选择: 使用扩展名,我们来看一下spring中如何配置这部分.



二.spring rest配置

现spring完成内容协商(content negotiation)的工作是由ContentNegotiatingViewResolver来完成的.它的工作模式支持我上面讲的三种, 
ContentNegotiatingViewResolver是根据客户提交的MimeType(如 text/html,application/xml)来跟服务端的一组viewResover的MimeType相比较,如果符合,即返回viewResover的数据.
而 /user/123.xml, ContentNegotiatingViewResolver会首先将 .xml 根据mediaTypes属性将其转换成 application/xml,然后完成前面所说的比较.


下面是ContentNegotiatingViewResolver的完全配置.


 


    
<!--  根据客户端的不同的请求决定不同的view进行响应, 如 /blog/1.json /blog/1.xml  -->
    
< bean  class ="org.springframework.web.servlet.view.ContentNegotiatingViewResolver" >
                
<!--  设置为true以忽略对Accept Header的支持 -->
                
< property  name ="ignoreAcceptHeader"  value ="true" />
                
<!--  在没有扩展名时即: "/user/1" 时的默认展现形式  -->
        
< property  name ="defaultContentType"  value ="text/html" />
        
                
<!--  扩展名至mimeType的映射,即 /user.json => application/json  -->
        
< property  name ="mediaTypes" >
            
< map >
                
< entry  key ="json"  value ="application/json"   />
                
< entry  key ="xml"  value ="application/xml"   />
            
</ map >
        
</ property >
        
<!--  用于开启 /userinfo/123?format=json 的支持  -->
        
< property  name ="favorParameter"  value ="false" />
        
< property  name ="viewResolvers" >
            
< list >
                
< bean  class ="org.springframework.web.servlet.view.BeanNameViewResolver"   />
                
< bean  class ="org.springframework.web.servlet.view.InternalResourceViewResolver" >
                    
< property  name ="viewClass"  value ="org.springframework.web.servlet.view.JstlView" />
                    
< property  name ="prefix"  value ="/pages" />
                    
< property  name ="suffix"  value =".jsp" ></ property >
                
</ bean >
            
</ list >
        
</ property >
        
< property  name ="defaultViews" >
            
< list >
                
<!--  for application/json  -->
                
< bean  class ="org.springframework.web.servlet.view.json.MappingJacksonJsonView"   />
                
<!--  for application/xml  -->
                
<!--  
                <bean class="org.springframework.web.servlet.view.xml.MarshallingView" >
                    <property name="marshaller">
                        <bean class="org.springframework.oxm.xstream.XStreamMarshaller"/>
                    </property>
                </bean>
                 
-->
            
</ list >
        
</ property >
    
</ bean >

 Spring MVC 之 ContentNegotiatingViewResolver(ajax+json格式的输入和输出)

Spring mvc处理json需要使用jackson的类库,因此为支持json格式的输入输出需要先修改pom.xml增加jackson包的引用 

Xml代码   收藏代码
  1. <!-- json -->  
  2. <dependency>  
  3.     <groupId>org.codehaus.jackson</groupId>  
  4.     <artifactId>jackson-core-lgpl</artifactId>  
  5.     <version>1.8.1</version>  
  6. </dependency>  
  7.   
  8. <dependency>  
  9.     <groupId>org.codehaus.jackson</groupId>  
  10.     <artifactId>jackson-mapper-lgpl</artifactId>  
  11.     <version>1.8.1</version>  
  12. </dependency>  




先修改之前的helloworld.jsp,增加客户端json格式的数据输入。 

Javascript代码   收藏代码
  1.     var cfg =   {  
  2.         type: 'POST',   
  3.         data: JSON.stringify({userName:'winzip',password:'password',mobileNO:'13818881888'}),   
  4.         dataType: 'json',  
  5.         contentType:'application/json;charset=UTF-8',         
  6.         success: function(result) {   
  7.             alert(result.success);   
  8.         }   
  9.     };  
  10.   
  11. function doTestJson(actionName){  
  12.     cfg.url = actionName;  
  13.     $.ajax(cfg);  
  14. }  




根据前面的分析,在spring mvc中解析输入为json格式的数据有两种方式 
1:使用@RequestBody来设置输入 

Java代码   收藏代码
  1.     @RequestMapping("/json1")  
  2.     @ResponseBody  
  3.     public JsonResult testJson1(@RequestBody User u){  
  4.         log.info("get json input from request body annotation");  
  5.         log.info(u.getUserName());  
  6.         return new JsonResult(true,"return ok");  
  7. }  



2:使用HttpEntity来实现输入绑定 

Java代码   收藏代码
  1.     @RequestMapping("/json2")      
  2.     public ResponseEntity<JsonResult> testJson2(HttpEntity<User> u){  
  3.         log.info("get json input from HttpEntity annotation");  
  4.         log.info(u.getBody().getUserName());  
  5.         ResponseEntity<JsonResult> responseResult = new ResponseEntity<JsonResult>( new JsonResult(true,"return ok"),HttpStatus.OK);  
  6.         return responseResult;  
  7. }  



Json格式的输出也对应有两种方式 
1:使用@responseBody来设置输出内容为context body 
2:返回值设置为ResponseEntity<?>类型,以返回context body 
另外,第三种方式是使用ContentNegotiatingViewResolver来设置输出为json格式,需要修改servlet context配置文件如下 

Xml代码   收藏代码
  1. <bean  
  2.     class="org.springframework.web.servlet.view.ContentNegotiatingViewResolver">  
  3.     <property name="order" value="1" />  
  4.     <property name="mediaTypes">  
  5.         <map>  
  6.             <entry key="json" value="application/json" />  
  7.         </map>  
  8.     </property>  
  9.     <property name="defaultViews">  
  10.         <list>  
  11.             <bean  
  12.                 class="org.springframework.web.servlet.view.json.MappingJacksonJsonView" />  
  13.         </list>  
  14.     </property>  
  15.     <property name="ignoreAcceptHeader" value="true" />  
  16. </bean>  


但这种格式的输出会返回{model类名:{内容}} 的json格式, 
例如,以下代码 

Java代码   收藏代码
  1. @RequestMapping("/json3.json")  
  2. public JsonResult testJson3(@RequestBody User u){  
  3.     log.info("handle json output from ContentNegotiatingViewResolver");  
  4.     return new JsonResult(true,"return ok");  
  5. }  


期望的返回是 {success:true,message:”return ok”}; 
但实际返回的却是 {"jsonResult":{"success":true,"msg":"return ok"}} 
原因是MappingJacksonJsonView中对返回值的处理未考虑modelMap中只有一个值的情况,直接是按照mapName:{mapResult}的格式来返回数据的。 
修改方法,重载MappingJacksonJsonView类并重写filterModel方法如下 

Java代码   收藏代码
  1. protected Object filterModel(Map<String, Object> model) {    
  2.     Map<?, ?> result = (Map<?, ?>) super.filterModel(model);    
  3.     if (result.size() == 1) {    
  4.         return result.values().iterator().next();    
  5.     } else {    
  6.         return result;    
  7.     }    
  8. }    



对应的ContentNegotiatingViewResolver修改如下 

Xml代码   收藏代码
  1. <bean  
  2.         class="org.springframework.web.servlet.view.ContentNegotiatingViewResolver">  
  3.         <property name="order" value="1" />  
  4.         <property name="mediaTypes">  
  5.             <map>  
  6.                 <entry key="json" value="application/json" />  
  7.             </map>  
  8.         </property>  
  9.         <property name="defaultViews">  
  10.             <list>  
  11.                 <bean  
  12.                     class="net.zhepu.json.MappingJacksonJsonView" />  
  13.             </list>  
  14.         </property>  
  15.         <property name="ignoreAcceptHeader" value="true" />  
  16.     </bean>  

你可能感兴趣的:(spring,mvc,REST)