spring 3.0 应用springmvc 构造RESTful URL 详细讲解

本文详细讲解了如何使用Spring MVC构建RESTful API,包括如何构造无扩展名的URL,解决与静态资源映射冲突的问题,以及处理浏览器不支持的HTTP DELETE和PUT请求。还介绍了配置URL重写过滤器和隐藏HTTP方法过滤器,以及Controller的编写方法。最后,探讨了内容协商在RESTful服务中的应用,并展示了如何根据不同的请求返回XML或JSON格式的数据。
摘要由CSDN通过智能技术生成

(一)

简单例子如下,比如如下URL

Java代码 复制代码  收藏代码
  1. /blog/1  HTTP GET =>    得到id = 1的blog   
  2. /blog/1  HTTP DELETE => 删除 id = 1的blog   
  3. /blog/1  HTTP PUT  =>   更新id = 1的blog   
  4. /blog     HTTP POST =>   新增BLOG  
/blog/1  HTTP GET =>    得到id = 1的blog
/blog/1  HTTP DELETE => 删除 id = 1的blog
/blog/1  HTTP PUT  =>   更新id = 1的blog
/blog     HTTP POST =>   新增BLOG

 

 

以下详细解一下spring rest使用.

 

首先,我们带着如下三个问题查看本文。
1. 如何在java构造没有扩展名的RESTful url,如 /forms/1,而不是 /forms/1.do

2. 由于我们要构造没有扩展名的url本来是处理静态资源的容器映射的,现在被我们的spring占用了,冲突怎么解决?
3. 浏览器的form标签不支持提交delete,put请求,如何曲线解决?

 

springmvc rest 实现


springmvc的resturl是通过@RequestMapping 及@PathVariable annotation提供的,通过如@RequestMapping(value="/blog/{id}",method=RequestMethod.DELETE)即可处理/blog/1 的delete请求.

Java代码 复制代码  收藏代码
  1. @RequestMapping(value="/blog/{id}",method=RequestMethod.DELETE)   
  2. public ModelAndView delete(@PathVariable Long id,HttpServletRequest request,HttpServletResponse response) {   
  3.     blogManager.removeById(id);   
  4.     return new ModelAndView(LIST_ACTION);   
  5. }  
@RequestMapping(value="/blog/{id}",method=RequestMethod.DELETE)
public ModelAndView delete(@PathVariable Long id,HttpServletRequest request,HttpServletResponse response) {
	blogManager.removeById(id);
	return new ModelAndView(LIST_ACTION);
}

 

@RequestMapping @PathVariable如果URL中带参数,则配合使用,如

Java代码 复制代码  收藏代码
  1. @RequestMapping(value="/blog/{blogId}/message/{msgId}",method=RequestMethod.DELETE)   
  2. public ModelAndView delete(@PathVariable("blogId") Long blogId,@PathVariable("msgId") Long msgId,HttpServletRequest request,HttpServletResponse response) {   
  3. }  
@RequestMapping(value="/blog/{blogId}/message/{msgId}",method=RequestMethod.DELETE)
public ModelAndView delete(@PathVariable("blogId") Long blogId,@PathVariable("msgId") Long msgId,HttpServletRequest request,HttpServletResponse response) {
}

 

 spring rest配置指南

1. springmvc web.xml配置

Xml代码 复制代码  收藏代码
  1. <!-- 该servlet为tomcat,jetty等容器提供,将静态资源映射从/改为/static/目录,如原来访问 http://localhost/foo.css ,现在http://localhost/static/foo.css -->  
  2. <servlet-mapping>  
  3.     <servlet-name>default</servlet-name>  
  4.     <url-pattern>/static/*</url-pattern>  
  5. </servlet-mapping>  
  6. <servlet>  
  7.     <servlet-name>springmvc</servlet-name>  
  8.     <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>  
  9.     <load-on-startup>1</load-on-startup>  
  10. </servlet>  
  11.   
  12. <!-- URL重写filter,用于将访问静态资源http://localhost/foo.css 转为http://localhost/static/foo.css -->  
  13. <filter>  
  14.     <filter-name>UrlRewriteFilter</filter-name>  
  15.     <filter-class>org.tuckey.web.filters.urlrewrite.UrlRewriteFilter</filter-class>  
  16.     <init-param>  
  17.             <param-name>confReloadCheckInterval</param-name>  
  18.             <param-value>60</param-value>  
  19.         </init-param>  
  20.     <init-param>  
  21.                 <param-name>logLevel</param-name>  
  22.                 <param-value>DEBUG</param-value>  
  23.         </init-param>        
  24. </filter>  
  25. <filter-mapping>  
  26.     <filter-name>UrlRewriteFilter</filter-name>  
  27.     <url-pattern>/*</url-pattern>  
  28. </filter-mapping>  
  29.   
  30. <!-- 覆盖default servlet的/, springmvc servlet将处理原来处理静态资源的映射 -->  
  31. <servlet-mapping>  
  32.     <servlet-name>springmvc</servlet-name>  
  33.     <url-pattern>/</url-pattern>  
  34. </servlet-mapping>  
  35.   
  36. <!-- 浏览器不支持put,delete等method,由该filter将/blog?_method=delete转换为标准的http delete方法 -->  
  37. <filter>  
  38.     <filter-name>HiddenHttpMethodFilter</filter-name>  
  39.     <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>  
  40. </filter>  
  41.   
  42. <filter-mapping>  
  43.     <filter-name>HiddenHttpMethodFilter</filter-name>  
  44.     <servlet-name>springmvc</servlet-name>  
  45. </filter-mapping>  
	<!-- 该servlet为tomcat,jetty等容器提供,将静态资源映射从/改为/static/目录,如原来访问 http://localhost/foo.css ,现在http://localhost/static/foo.css -->
	<servlet-mapping>
		<servlet-name>default</servlet-name>
		<url-pattern>/static/*</url-pattern>
	</servlet-mapping>
	<servlet>
	    <servlet-name>springmvc</servlet-name>
	    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
	    <load-on-startup>1</load-on-startup>
	</servlet>
	
	<!-- URL重写filter,用于将访问静态资源http://localhost/foo.css 转为http://localhost/static/foo.css -->
	<filter>
		<filter-name>UrlRewriteFilter</filter-name>
		<filter-class>org.tuckey.web.filters.urlrewrite.UrlRewriteFilter</filter-class>
		<init-param>
		    	<param-name>confReloadCheckInterval</param-name>
		    	<param-value>60</param-value>
    		</init-param>
		<init-param>
            		<param-name>logLevel</param-name>
            		<param-value>DEBUG</param-value>
        	</init-param>    	
	</filter>
	<filter-mapping>
		<filter-name>UrlRewriteFilter</filter-name>
		<url-pattern>/*</url-pattern>
	</filter-mapping>
	
	<!-- 覆盖default servlet的/, springmvc servlet将处理原来处理静态资源的映射 -->
	<servlet-mapping>
	    <servlet-name>springmvc</servlet-name>
	    <url-pattern>/</url-pattern>
	</servlet-mapping>
	
	<!-- 浏览器不支持put,delete等method,由该filter将/blog?_method=delete转换为标准的http delete方法 -->
	<filter>
		<filter-name>HiddenHttpMethodFilter</filter-name>
		<filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
	</filter>
	
	<filter-mapping>
		<filter-name>HiddenHttpMethodFilter</filter-name>
		<servlet-name>springmvc</servlet-name>
	</filter-mapping>

 

 

2. webapp/WEB-INF/springmvc-servlet.xml配置,使用如下两个class激活@RequestMapping annotation

Java代码 复制代码  收藏代码
  1. <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"/>   
  2. <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter"/>  
    <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"/>
    <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter"/>

 

完整配置

Java代码 复制代码  收藏代码
  1. <beans default-autowire="byName"   >   
  2.   
  3.     <!-- 自动搜索@Controller标注的类 -->   
  4.     <context:component-scan base-package="com.**.controller"/>   
  5.        
  6.     <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"/>   
  7.   
  8.     <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter"/>   
  9.   
  10.     <!-- Default ViewResolver -->   
  11.     <bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">   
  12.         <property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/>   
  13.         <property name="prefix" value="/pages"/>   
  14.         <property name="suffix" value=".jsp"></property>   
  15.     </bean>   
  16.        
  17.     <bean id="messageSource" class="org.springframework.context.support.ResourceBundleMessageSource" p:basename="i18n/messages"/>   
  18.   
  19.     <!-- Mapping exception to the handler view -->   
  20.     <bean id="exceptionResolver" class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">   
  21.         <!-- to /commons/error.jsp -->   
  22.         <property name="defaultErrorView" value="/commons/error"/>   
  23.         <property name="exceptionMappings">   
  24.             <props>   
  25.             </props>   
  26.         </property>   
  27.     </bean>   
  28.            
  29. </beans>  
<beans default-autowire="byName"   >

	<!-- 自动搜索@Controller标注的类 -->
	<context:component-scan base-package="com.**.controller"/>
	
    <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"/>

    <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter"/>

    <!-- Default ViewResolver -->
    <bean id="viewResolver" 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>
    
    <bean id="messageSource" class="org.springframework.context.support.ResourceBundleMessageSource" p:basename="i18n/messages"/>

    <!-- Mapping exception to the handler view -->
    <bean id="exceptionResolver" class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
    	<!-- to /commons/error.jsp -->
        <property name="defaultErrorView" value="/commons/error"/>
        <property name="exceptionMappings">
            <props>
            </props>
        </property>
    </bean>
        
</beans>

 

 

3. Controller编写

Java代码 复制代码  收藏代码
  1. /**  
  2.  * @RequestMapping("/userinfo") 具有层次关系,方法级的将在类一级@RequestMapping之一,  
  3.  * 如下面示例, 访问方法级别的@RequestMapping("/new"),则URL为 /userinfo/new  
  4.  */  
  5. @Controller  
  6. @RequestMapping("/userinfo")   
  7. public class UserInfoController extends BaseSpringController{   
  8.     //默认多列排序,example: username desc,createTime asc   
  9.     protected static final String DEFAULT_SORT_COLUMNS = null;    
  10.        
  11.     private UserInfoManager userInfoManager;   
  12.        
  13.     private final String LIST_ACTION = "redirect:/userinfo";   
  14.        
  15.     /**   
  16.      * 通过spring自动注入  
  17.      **/  
  18.     public void setUserInfoManager(UserInfoManager manager) {   
  19.         this.userInfoManager = manager;   
  20.     }   
  21.        
  22.     /** 列表 */  
  23.     @RequestMapping  
  24.     public ModelAndView index(HttpServletRequest request,HttpServletResponse response,UserInfo userInfo) {   
  25.         PageRequest<Map> pageRequest = newPageRequest(request,DEFAULT_SORT_COLUMNS);   
  26.         //pageRequest.getFilters(); //add custom filters   
  27.            
  28.         Page page = this.userInfoManager.findByPageRequest(pageRequest);   
  29.         savePage(page,pageRequest,request);   
  30.         return new ModelAndView("/userinfo/list","userInfo",userInfo);   
  31.     }   
  32.        
  33.     /** 进入新增 */  
  34.     @RequestMapping(value="/new")   
  35.     public ModelAndView _new(HttpServletRequest request,HttpServletResponse response,UserInfo userInfo) throws Exception {   
  36.         return new ModelAndView("/userinfo/new","userInfo",userInfo);   
  37.     }   
  38.        
  39.     /** 显示 */  
  40.     @RequestMapping(value="/{id}")   
  41.     public ModelAndView show(@PathVariable Long id,HttpServletRequest request,HttpServletResponse response) throws Exception {   
  42.         UserInfo userInfo = (UserInfo)userInfoManager.getById(id);   
  43.         return new ModelAndView("/userinfo/show","userInfo",userInfo);   
  44.     }   
  45.        
  46.     /** 编辑 */  
  47.     @RequestMapping(value="/{id}/edit")   
  48.     public ModelAndView edit(@PathVariable Long id,HttpServletRequest request,HttpServletResponse response) throws Exception {   
  49.         UserInfo userInfo = (UserInfo)userInfoManager.getById(id);   
  50.         return new ModelAndView("/userinfo/edit","userInfo",userInfo);   
  51.     }   
  52.        
  53.     /** 保存新增 */  
  54.     @RequestMapping(method=RequestMethod.POST)   
  55.     public ModelAndView create(HttpServletRequest request,HttpServletResponse response,UserInfo userInfo) throws Exception {   
  56.         userInfoManager.save(userInfo);   
  57.         return new ModelAndView(LIST_ACTION);   
  58.     }   
  59.        
  60.     /** 保存更新 */  
  61.     @RequestMapping(value="/{id}",method=RequestMethod.PUT)   
  62.     public ModelAndView update(@PathVariable Long id,HttpServletRequest request,HttpServletResponse response) throws Exception {   
  63.         UserInfo userInfo = (UserInfo)userInfoManager.getById(id);   
  64.         bind(request,userInfo);   
  65.         userInfoManager.update(userInfo);   
  66.         return new ModelAndView(LIST_ACTION);   
  67.     }   
  68.        
  69.     /** 删除 */  
  70.     @RequestMapping(value="/{id}",method=RequestMethod.DELETE)   
  71.     public ModelAndView delete(@PathVariable Long id,HttpServletRequest request,HttpServletResponse response) {   
  72.         userInfoManager.removeById(id);   
  73.         return new ModelAndView(LIST_ACTION);   
  74.     }   
  75.   
  76.     /** 批量删除 */  
  77.     @RequestMapping(method=RequestMethod.DELETE)   
  78.     public ModelAndView batchDelete(@RequestParam("items") Long[] items,HttpServletRequest request,HttpServletResponse response) {   
  79.            
  80.         for(int i = 0; i < items.length; i++) {   
  81.                
  82.             userInfoManager.removeById(items[i]);   
  83.         }   
  84.         return new ModelAndView(LIST_ACTION);   
  85.     }   
  86.        
  87. }  
/**
 * @RequestMapping("/userinfo") 具有层次关系,方法级的将在类一级@RequestMapping之一,
 * 如下面示例, 访问方法级别的@RequestMapping("/new"),则URL为 /userinfo/new
 */
@Controller
@RequestMapping("/userinfo")
public class UserInfoController extends BaseSpringController{
	//默认多列排序,example: username desc,createTime asc
	protected static final String DEFAULT_SORT_COLUMNS = null; 
	
	private UserInfoManager userInfoManager;
	
	private final String LIST_ACTION = "redirect:/userinfo";
	
	/** 
	 * 通过spring自动注入
	 **/
	public void setUserInfoManager(UserInfoManager manager) {
		this.userInfoManager = manager;
	}
	
	/** 列表 */
	@RequestMapping
	public ModelAndView index(HttpServletRequest request,HttpServletResponse response,UserInfo userInfo) {
		PageRequest<Map> pageRequest = newPageRequest(request,DEFAULT_SORT_COLUMNS);
		//pageRequest.getFilters(); //add custom filters
		
		Page page = this.userInfoManager.findByPageRequest(pageRequest);
		savePage(page,pageRequest,request);
		return new ModelAndView("/userinfo/list","userInfo",userInfo);
	}
	
	/** 进入新增 */
	@RequestMapping(value="/new")
	public ModelAndView _new(HttpServletRequest request,HttpServletResponse response,UserInfo userInfo) throws Exception {
		return new ModelAndView("/userinfo/new","userInfo",userInfo);
	}
	
	/** 显示 */
	@RequestMapping(value="/{id}")
	public ModelAndView show(@PathVariable Long id,HttpServletRequest request,HttpServletResponse response) throws Exception {
		UserInfo userInfo = (UserInfo)userInfoManager.getById(id);
		return new ModelAndView("/userinfo/show","userInfo",userInfo);
	}
	
	/** 编辑 */
	@RequestMapping(value="/{id}/edit")
	public ModelAndView edit(@PathVariable Long id,HttpServletRequest request,HttpServletResponse response) throws Exception {
		UserInfo userInfo = (UserInfo)userInfoManager.getById(id);
		return new ModelAndView("/userinfo/edit","userInfo",userInfo);
	}
	
	/** 保存新增 */
	@RequestMapping(method=RequestMethod.POST)
	public ModelAndView create(HttpServletRequest request,HttpServletResponse response,UserInfo userInfo) throws Exception {
		userInfoManager.save(userInfo);
		return new ModelAndView(LIST_ACTION);
	}
	
	/** 保存更新 */
	@RequestMapping(value="/{id}",method=RequestMethod.PUT)
	public ModelAndView update(@PathVariable Long id,HttpServletRequest request,HttpServletResponse response) throws Exception {
		UserInfo userInfo = (UserInfo)userInfoManager.getById(id);
		bind(request,userInfo);
		userInfoManager.update(userInfo);
		return new ModelAndView(LIST_ACTION);
	}
	
	/** 删除 */
	@RequestMapping(value="/{id}",method=RequestMethod.DELETE)
	public ModelAndView delete(@PathVariable Long id,HttpServletRequest request,HttpServletResponse response) {
		userInfoManager.removeById(id);
		return new ModelAndView(LIST_ACTION);
	}

	/** 批量删除 */
	@RequestMapping(method=RequestMethod.DELETE)
	public ModelAndView batchDelete(@RequestParam("items") Long[] items,HttpServletRequest request,HttpServletResponse response) {
		
		for(int i = 0; i < items.length; i++) {
			
			userInfoManager.removeById(items[i]);
		}
		return new ModelAndView(LIST_ACTION);
	}
	
}

 

 

上面是rapid-framework 新版本生成器生成的代码,以后也将应用此规则,rest url中增删改查等基本方法与Controller的方法映射规则

Java代码 复制代码  收藏代码
  1. /userinfo           => index()   
  2. /userinfo/new       => _new()   
  3. /userinfo/{id}      => show()   
  4. /userinfo/{id}/edit         => edit()   
  5. /userinfo   POST        => create()   
  6. /userinfo/{id}  PUT => update()   
  7. /userinfo/{id}  DELETE  => delete()   
  8. /userinfo   DELETE      => batchDelete()  
	/userinfo 			=> index()
	/userinfo/new		=> _new()
	/userinfo/{id}		=> show()
	/userinfo/{id}/edit 		=> edit()
	/userinfo 	POST		=> create()
	/userinfo/{id} 	PUT	=> update()
	/userinfo/{id} 	DELETE	=> delete()
	/userinfo 	DELETE		=> batchDelete()

 注(不使用 /userinfo/add  => add() 方法是由于add这个方法会被maxthon浏览器当做广告链接过滤掉,因为包含ad字符)

 

4. jsp 编写

Html代码 复制代码  收藏代码
  1. <form:form action="${ctx}/userinfo/${userInfo.userId}" method="put">  
  2. </form:form>  
<form:form action="${ctx}/userinfo/${userInfo.userId}" method="put">
</form:form>

 生成的html内容如下, 生成一个hidden的_method=put,并于web.xml中的HiddenHttpMethodFilter配合使用,在服务端将post请求改为put请求

Java代码 复制代码  收藏代码
  1. <form id="userInfo" action="/springmvc_rest_demo/userinfo/2" method="post">   
  2.     <input type="hidden" name="_method" value="put"/>   
  3. </form>  
<form id="userInfo" action="/springmvc_rest_demo/userinfo/2" method="post">
	<input type="hidden" name="_method" value="put"/>
</form>

 

另外一种方法是你可以使用ajax发送put,delete请求.

 

5. 静态资源的URL重写

   如上我们描述,现因为将default servlet映射至/static/的子目录,现我们访问静态资源将会带一个/static/前缀.

   如 /foo.gif, 现在访问该文件将是 /static/foo.gif.
   那如何避免这个前缀呢,那就是应用URL rewrite,现我们使用 http://tuckey.org/urlrewrite/, 重写规则如下

 

Xml代码 复制代码  收藏代码
  1. <urlrewrite>  
  2.     <!-- 访问jsp及jspx将不rewrite url,其它.js,.css,.gif等将重写,如 /foo.gif => /static/foo.gif -->  
  3.     <rule>  
  4.         <condition operator="notequal" next="and" type="request-uri">.*.jsp</condition>  
  5.         <condition operator="notequal" next="and" type="request-uri">.*.jspx</condition>  
  6.         <from>^(/.*\..*)$</from>  
  7.         <to>/static$1</to>  
  8.     </rule>  
  9. </urlrewrite>  
<urlrewrite>
    <!-- 访问jsp及jspx将不rewrite url,其它.js,.css,.gif等将重写,如 /foo.gif => /static/foo.gif -->
    <rule>
    	<condition operator="notequal" next="and" type="request-uri">.*.jsp</condition>
    	<condition operator="notequal" next="and" type="request-uri">.*.jspx</condition>
        <from>^(/.*\..*)$</from>
        <to>/static$1</to>
    </rule>
</urlrewrite>

   另笔者专门写了一个 RestUrlRewriteFilter来做同样的事件,以后会随着rapid-framework一起发布. 比这个更加轻量级.

 

并且该代码已经贡献给spring,不知会不会在下一版本发布

 

 

(二)

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


1.使用http request header: Accept
Xml代码 复制代码  收藏代码
  1. GET /user/123 HTTP/1.1   
  2. Accept: application/xml                 //将返回xml格式数据   
  3.   
  4. GET /user/123 HTTP/1.1   
  5. Accept: application/json               //将返回json格式数据  
GET /user/123 HTTP/1.1
Accept: application/xml                 //将返回xml格式数据

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

2.使用扩展名
Html代码 复制代码  收藏代码
  1. /user/123.xml  将返回xml格式数据   
  2. /user/123.json 将返回json格式数据   
  3. /user/123.html 将返回html格式数据  
/user/123.xml  将返回xml格式数据
/user/123.json 将返回json格式数据
/user/123.html 将返回html格式数据
 

3.使用参数
Html代码 复制代码  收藏代码
  1. /user/123?format=xml          //将返回xml数据   
  2. /user/123?format=json          //将返回json数据  
/user/123?format=xml          //将返回xml数据
/user/123?format=json          //将返回json数据
 

而以上三种各有优缺点:
1.使用Accept header:
   这一种为教科书中通常描述的一种,理想中这种方式也是最好的,但如果你的资源要给用户直接通过浏览器访问(即html展现),那么由于浏览器的差异,发送上来的Accept Header头将是不一样的. 将导致服务器不知要返回什么格式的数据给你. 下面是浏览器的Accept Header
Html代码 复制代码  收藏代码
  1. chrome:   
  2. Accept:application/xml,application/xhtml+xml,textml;q=0.9,text/plain;q=0.8,image/png,*/*;q=0.5   
  3.   
  4. firefox:   
  5. Accept:text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8   
  6.   
  7. IE8:   
  8. 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, */*  
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.使用参数
  现在很多open API是使用这种方式,但可能由于要编写的字符较多,所以较少使用.

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

二.spring配置

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

下面是ContentNegotiatingViewResolver的完全配置.

Xml代码 复制代码  收藏代码
  1.   <!-- 根据客户端的不同的请求决定不同的view进行响应, 如 /blog/1.json /blog/1.xml -->  
  2.   <bean class="org.springframework.web.servlet.view.ContentNegotiatingViewResolver">  
  3.               <!-- 设置为true以忽略对Accept Header的支持-->  
  4.             <property name="ignoreAcceptHeader" value="true"/>  
  5.               <!-- 在没有扩展名时即: "/user/1" 时的默认展现形式 -->  
  6. <property name="defaultContentType" value="text/html"/>  
  7.   
  8.             <!-- 扩展名至mimeType的映射,即 /user.json => application/json -->  
  9. <property name="mediaTypes">  
  10.     <map>  
  11.         <entry key="json" value="application/json" />  
  12.         <entry key="xml" value="application/xml" />  
  13.     </map>  
  14. </property>  
  15. <!-- 用于开启 /userinfo/123?format=json 的支持 -->  
  16. <property name="favorParameter" value="false"/>  
  17. <property name="viewResolvers">  
  18.     <list>  
  19.         <bean class="org.springframework.web.servlet.view.BeanNameViewResolver" />  
  20.         <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">  
  21.             <property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/>  
  22.             <property name="prefix" value="/pages"/>  
  23.             <property name="suffix" value=".jsp"></property>  
  24.         </bean>  
  25.     </list>  
  26. </property>  
  27. <property name="defaultViews">  
  28.     <list>  
  29.         <!-- for application/json -->  
  30.         <bean class="org.springframework.web.servlet.view.json.MappingJacksonJsonView" />  
  31.         <!-- for application/xml -->  
  32.         <!--    
  33.         <bean class="org.springframework.web.servlet.view.xml.MarshallingView" >  
  34.             <property name="marshaller">  
  35.                 <bean class="org.springframework.oxm.xstream.XStreamMarshaller"/>  
  36.             </property>  
  37.         </bean>  
  38.          -->  
  39.     </list>  
  40. </property>  
  41.   </bean>  
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值