SpringMVC配置多视图-内容协商原理

2014年03月06日 16:46:59 日积月累_滴水石穿 阅读数:10964更多

个人分类: SpringMVC
 

Spring Framework 3.2增强了ContentNegotiationManager,使得配置多视图变得尤为轻松。并且对于多视图的解析的实现都可以有多种供你选择。如果你想使用Spring作为网站后台,并且想完全分离 前后台的代码依赖,那么了解如何配置Spring的基于内容协商多视图是非常必须而且有用的。下面就来看看如何配置Spring,让它支持JSON/XML视图吧。

先看看Spring的官方文档关于Content Negotiation的improvements

 
A ContentNeogtiationStrategy is now available for resolving the requested media types from an incoming request. The available implementations are based on the file extension, query parameter, the ‘Accept’ header, or a fixed content type. Equivalent options were previously available only in the ContentNegotiatingViewResolver but are now available throughout. ContentNegotiationManager is the central class to use when configuring content negotiation options. For more details see Section 17.15.4, “Configuring Content Negotiation”. The introduction of ContentNegotiationManger also enables selective suffix pattern matching for incoming requests. For more details, see the Javadoc ofRequestMappingHandlerMapping.setUseRegisteredSuffixPatternMatch.
一个ContentNeogtiationStrategy(是spring官方拼写错了?)现在可以使用了,它可用于解决传入请求所请求的媒体类型。可用的实现有基于文件扩展名(后缀)、请求参数、“Accept”头、或者某一固定的类容类型。与之前可以使用在ContentNegotiatingViewResolver的配置项是等效的。ContentNegotiationManager是用来配置内容协商选项的中心类。更多细节见17.15.4节 “配置内容协商”。ContentNegotiationManager的引进可以使用可选的后缀模式来匹配(映射)传入的请求,更多细节请查看RequestMappingHandlerMapping.setUserRegisteredSuffixPatternMatch.

可见,内容协商其实说白了很简单,就是根据请求规则决定返回什么样的内容类型。后缀规则、参数规则、Accept头规则、固定的内容类型等。注意,这里只是决定,不是具体提供内容类型的地方。

好了,现在正式开始配置的介绍:

这里默认你已经配置好Spring的DispatcherServlet,并设置映射路径是“/”,例如下面的配置:

  1.  
    <?xml version="1.0" encoding="UTF-8"?>
  2.  
    <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3.  
    xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
  4.  
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
  5.  
    id="WebApp_ID" version="3.0">
  6.  
    <display-name>lab-order</display-name>
  7.  
    <welcome-file-list>
  8.  
    <welcome-file>index.html</welcome-file>
  9.  
    <welcome-file>index.htm</welcome-file>
  10.  
    <welcome-file>index.jsp</welcome-file>
  11.  
    <welcome-file>default.html</welcome-file>
  12.  
    <welcome-file>default.htm</welcome-file>
  13.  
    <welcome-file>default.jsp</welcome-file>
  14.  
    </welcome-file-list>
  15.  
     
  16.  
    <context-param>
  17.  
    <description>上下文配置地址</description>
  18.  
    <param-name>contextConfigLocation</param-name>
  19.  
    <param-value>/WEB-INF/mvc-servlet.xml</param-value>
  20.  
    </context-param>
  21.  
     
  22.  
    <context-param>
  23.  
    <description>log4j配置位置</description>
  24.  
    <param-name>log4jConfigLocation</param-name>
  25.  
    <param-value>/WEB-INF/log4j.properties</param-value>
  26.  
    </context-param>
  27.  
     
  28.  
    <servlet>
  29.  
    <description>核心Servlet</description>
  30.  
    <servlet-name>mvc</servlet-name>
  31.  
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  32.  
    </servlet>
  33.  
    <servlet-mapping>
  34.  
    <servlet-name>mvc</servlet-name>
  35.  
    <url-pattern>/</url-pattern>
  36.  
    </servlet-mapping>
  37.  
     
  38.  
    <listener>
  39.  
    <description>log4j配置侦听</description>
  40.  
    <listener-class>org.springframework.web.util.Log4jConfigListener</listener-class>
  41.  
    </listener>
  42.  
     
  43.  
    <listener>
  44.  
    <description>Spring上下文侦听加载器</description>
  45.  
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  46.  
    </listener>
  47.  
     
  48.  
    <filter>
  49.  
    <description>编码过滤器</description>
  50.  
    <filter-name>encodingFilter</filter-name>
  51.  
    <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
  52.  
    <init-param>
  53.  
    <param-name>encoding</param-name>
  54.  
    <param-value>UTF-8</param-value>
  55.  
    </init-param>
  56.  
    <init-param>
  57.  
    <param-name>forceEncoding</param-name>
  58.  
    <param-value>true</param-value>
  59.  
    </init-param>
  60.  
    </filter>
  61.  
    <filter-mapping>
  62.  
    <filter-name>encodingFilter</filter-name>
  63.  
    <url-pattern>/*</url-pattern>
  64.  
    </filter-mapping>
  65.  
     
  66.  
    </web-app>
 
主要关注点在于核心DespatcherSevlet的映射路径,是“/”而不是“*.do”。

然后配置核心Sevlet的上下文环境,这里是文件“mvc-servlet.xml”。如下:

  1.  
    <beans xmlns="http://www.springframework.org/schema/beans"
  2.  
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
  3.  
    xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:p="http://www.springframework.org/schema/p"
  4.  
    xmlns:c="http://www.springframework.org/schema/c"
  5.  
    xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd
  6.  
    http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
  7.  
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
  8.  
     
  9.  
    <!-- 开启注解驱动 -->
  10.  
    <!-- 这样可以使用@Controller这些注解 -->
  11.  
    <mvc:annotation-driven
  12.  
    content-negotiation-manager="contentNegotiationManager"></mvc:annotation-driven>
  13.  
    <!-- 开启默认处理 -->
  14.  
    <!-- 这样静态资源就可以访问了 -->
  15.  
    <mvc:default-servlet-handler />
  16.  
     
  17.  
    <!-- 开启上下文注解配置 -->
  18.  
    <!-- 使@Autowire注解被支持 -->
  19.  
    <context:annotation-config></context:annotation-config>
  20.  
    <!-- 配置注解扫描包 -->
  21.  
    <context:component-scan base-package="org.laohu.modules"
  22.  
    annotation-config="true"></context:component-scan>
  23.  
     
  24.  
    <!-- 导入hibernate配置 -->
  25.  
    <import resource="hibernate-beans.xml" />
  26.  
    <!-- 导入视图解析配置 -->
  27.  
    <import resource="view-resolve.xml" />
  28.  
     
  29.  
    <!-- 导入模块配置 -->
  30.  
    <import resource="school-module.xml" />
  31.  
    </beans>
 
在“mvc-sevlet.xml”中并没有将所有的bean都放在里面,这是我的个人习惯,使用import指令(标签)将bean的配置分开配置,避免文件过大,修改或者查看比较麻烦,同时对于调试也是很有帮助的。

“mvc-servlet.xml”中

<mvc:annotation-driven content-negotiation-manager="contentNegotiationManager"></mvc:annotation-driven>

content-negotiation-manager属性就是指定内容协商管理器的bean,按照官方文档是这样的配置,但是有一个问题,这个问题稍后再和大家讨论。contentNegotiationManager这个bean配置在“view-resolve.xml中”:

  1.  
    <beans xmlns="http://www.springframework.org/schema/beans"
  2.  
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:c="http://www.springframework.org/schema/c"
  3.  
    xmlns:p="http://www.springframework.org/schema/p"
  4.  
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
  5.  
     
  6.  
    <!-- 配置内容协商视图解析 -->
  7.  
    <bean id="contentNegotiatingViewResolver"
  8.  
    class="org.springframework.web.servlet.view.ContentNegotiatingViewResolver">
  9.  
    <!-- 设置内容协商管理器 -->
  10.  
    <property name="contentNegotiationManager" ref="contentNegotiationManager"></property>
  11.  
    <!-- 设置默认视图 -->
  12.  
    <property name="defaultViews">
  13.  
    <list>
  14.  
    <ref bean="mappingJacksonJsonView" />
  15.  
    <ref bean="marshallingView" />
  16.  
    </list>
  17.  
    </property>
  18.  
    <!-- 设置视图解析器 -->
  19.  
    <property name="viewResolvers">
  20.  
    <list>
  21.  
    <ref bean="defalutViewResolver" />
  22.  
    </list>
  23.  
    </property>
  24.  
    </bean>
  25.  
     
  26.  
    <bean id="defalutViewResolver"
  27.  
    class="org.springframework.web.servlet.view.InternalResourceViewResolver">
  28.  
    <property name="prefix" value="/WEB-INF/view"></property>
  29.  
    <property name="suffix" value=".jsp"></property>
  30.  
    </bean>
  31.  
     
  32.  
    <!-- JSON视图 -->
  33.  
    <bean id="mappingJacksonJsonView"
  34.  
    class="org.springframework.web.servlet.view.json.MappingJacksonJsonView"></bean>
  35.  
    <!-- XML视图 -->
  36.  
    <bean id="marshallingView"
  37.  
    class="org.springframework.web.servlet.view.xml.MarshallingView">
  38.  
    <property name="marshaller">
  39.  
    <bean class="org.springframework.oxm.xstream.XStreamMarshaller">
  40.  
    <property name="supportedClasses">
  41.  
    <list>
  42.  
    <value>java.util.Collection</value>
  43.  
    <value>org.laohu.modules.school.model.School</value>
  44.  
    <value>org.laohu.modules.school.model.Laboratory</value>
  45.  
    <value>org.laohu.modules.school.model.stuff.LabMgr</value>
  46.  
    </list>
  47.  
    </property>
  48.  
    <property name="aliases">
  49.  
    <map>
  50.  
    <entry value="org.laohu.modules.school.model.School" key="school"></entry>
  51.  
    <entry value="org.laohu.modules.school.model.Laboratory"
  52.  
    key="laboratory"></entry>
  53.  
    <entry value="org.laohu.modules.school.model.stuff.LabMgr"
  54.  
    key="labmgr"></entry>
  55.  
    </map>
  56.  
    </property>
  57.  
    </bean>
  58.  
    </property>
  59.  
    </bean>
  60.  
     
  61.  
    <bean id="contentNegotiationManager"
  62.  
    class="org.springframework.web.accept.ContentNegotiationManagerFactoryBean">
  63.  
    <property name="mediaTypes">
  64.  
    <props>
  65.  
    <prop key="json">application/json</prop>
  66.  
    <prop key="xml">application/xml</prop>
  67.  
    </props>
  68.  
    </property>
  69.  
    </bean>
  70.  
     
  71.  
    </beans>

这里配置了多个bean,先从contentNegotiateManager说起,contentNegotiateManager并不是直接使用ContentNegotiateManager构造的,而是使用其工厂bean生产的,给他配置属性mediaTypes这个属性是告诉contentNegotiateManager将每一个mediaTypes里的entry作为文件名/URL后缀,其内容类型就是entry对应的value值。也就是说,如果请求的url为http://hostname/xxx/xxx/data.json?p1=2332&p2=abc的话,contentNegotiateManager就会认为你请求的内容类型(Content-Type)为application/json,那么它就要将响应的内容类型(Content-Type)设置为application/json;如下图:

为了方便大家理解,这里贴一下火狐的请求头和服务器的响应头:

  1.  
    GET /lab-order/school/.json HTTP/1.1
  2.  
    Host: localhost:8080
  3.  
    User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64; rv:19.0) Gecko/20100101 Firefox/19.0 FirePHP/0.7.1
  4.  
    Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
  5.  
    Accept-Language: zh-cn,zh;q=0.8,en-us;q=0.5,en;q=0.3
  6.  
    Accept-Encoding: gzip, deflate
  7.  
    Cookie: ext-mainnav.west=o%3Acolumns%3Da%253Ao%25253Aid%25253Ds%2525253Aheader-1041; ext-mainnav.east.description=o%3Acollapsed%3Ds%253Atop; ext-mainnav.east=o%3Awidth%3Dn%253A322; ext-stateGrid=o%3Awidth%3Dn%253A650%5Eheight%3Dn%253A350%5Ecolumns%3Da%253Ao%25253Aid%25253Ds%2525253Aheader-1247%255Eo%25253Aid%25253Ds%2525253Aheader-1248%255Eo%25253Aid%25253Ds%2525253Aheader-1249%255Eo%25253Aid%25253Ds%2525253Aheader-1250%255Eo%25253Aid%25253Ds%2525253Aheader-1251%255Eo%25253Aid%25253Ds%2525253Aheader-1252
  8.  
    x-insight: activate
  9.  
    Connection: keep-alive
  10.  
    Cache-Control: max-age=0
  1.  
    HTTP/1.1 200 OK
  2.  
    Server: Apache-Coyote/1.1
  3.  
    Pragma: no-cache
  4.  
    Cache-Control: no-cache, no-store, max-age=0
  5.  
    Expires: Thu, 01 Jan 1970 00:00:00 GMT
  6.  
    Content-Type: application/json;charset=UTF-8
  7.  
    Content-Language: zh-CN
  8.  
    Transfer-Encoding: chunked
  9.  
    Date: Thu, 04 Apr 2013 17:38:00 GMT

可以看到请求头是普通的text/html请求,但服务器相响应的是application/json的内容类型,表明内容协商工作正常,为了进一步测试内容协商管理器是否是按照这个基于后缀的映射内容类型,我们改变映射关系,修改mediaTypes:

  1.  
    <prop key="json">application/json</prop>
  2.  
    <prop key="xml">application/json</prop>

即将xml也映射到application/json上,再次使用.xml访问:

请求头:

  1.  
    GET /lab-order/school/.xml HTTP/1.1
  2.  
    Host: localhost:8080
  3.  
    User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64; rv:19.0) Gecko/20100101 Firefox/19.0 FirePHP/0.7.1
  4.  
    Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
  5.  
    Accept-Language: zh-cn,zh;q=0.8,en-us;q=0.5,en;q=0.3
  6.  
    Accept-Encoding: gzip, deflate
  7.  
    Cookie: ext-mainnav.west=o%3Acolumns%3Da%253Ao%25253Aid%25253Ds%2525253Aheader-1041; ext-mainnav.east.description=o%3Acollapsed%3Ds%253Atop; ext-mainnav.east=o%3Awidth%3Dn%253A322; ext-stateGrid=o%3Awidth%3Dn%253A650%5Eheight%3Dn%253A350%5Ecolumns%3Da%253Ao%25253Aid%25253Ds%2525253Aheader-1247%255Eo%25253Aid%25253Ds%2525253Aheader-1248%255Eo%25253Aid%25253Ds%2525253Aheader-1249%255Eo%25253Aid%25253Ds%2525253Aheader-1250%255Eo%25253Aid%25253Ds%2525253Aheader-1251%255Eo%25253Aid%25253Ds%2525253Aheader-1252
  8.  
    x-insight: activate
  9.  
    Connection: keep-alive

响应头:

  1.  
    HTTP/1.1 200 OK
  2.  
    Server: Apache-Coyote/1.1
  3.  
    Pragma: no-cache
  4.  
    Cache-Control: no-cache, no-store, max-age=0
  5.  
    Expires: Thu, 01 Jan 1970 00:00:00 GMT
  6.  
    Content-Type: application/json;charset=UTF-8
  7.  
    Content-Language: zh-CN
  8.  
    Transfer-Encoding: chunked
  9.  
    Date: Thu, 04 Apr 2013 17:50:55 GMT

响应正文:

{"schools":[{"id":1,"name":"江苏科技大学","position":"江苏镇江","content":30,"labMgrs":[],"laboratorys":[]}]}

以上就证明了当内容协商管理器使用后缀策略时的工作规律。

那么现在内容管理器知道了该响应给浏览器的内容类型后,该如何响应该内容类型给浏览器呢?contentNegotiationManager并不负责视图(数据如何呈现,JSON视图/XML视图等等),真正处理呈现的叫ViewResolver,视图解析器OR视图渲染器(姑且这么翻译),例如上面配置的defaultViewResolver就是默认的视图解析器他解析普通的jsp视图,这里不对它进行讨论,在稍后的文章中或许会专门讲一下它。但在ContentNegotiationViewResolver中配置的ViewResolver是在配置的defaultViews都没有匹配的时候才进行交接的。

那我们看看defaultViews都有些什么:mappingJacksonJsonView——传说中的JSON视图、marshallingView——编组视图XML视图。在defaultViews里注册的视图会在ContentNegotiationViewResolver中注册自己支持的内容类型,当contentNegotiationManager决定好响应的内容类型后,ContentNegotiationViewResolver就会根据该内容类型选择一个兼容的View进行渲染输出,当注册的内容类型都不兼容时,会查询viewResolver中的ViewResolver是否支持该请求,如果ViewResolver表示支持该请求,那么就由该ViewResolver负责视图渲染,如果ViewResolver表示不支持该请求,则查询下一个ViewResolver,直至所有的ViewResolver查询完毕。一旦有View对请求内容匹配,就直接渲染输出,不会进行ViewResolver的查询。由于这里配置了defaultViewResolver是InternalResourceViewResolver,它会对所有的请求说yes,所以这里的其他请求类型(非JSON/XML)都会交给它处理。查看Spring的类库,有不少ViewResolver的实现,有兴趣的同学可以去看看,我还没来得及细看这些实现,所以不会多讲这方面内容。

SpringMVC配置多视图-内容协商原理的更多相关文章

  1. SpringMVC整合freeMarker实现页面静态化+SpringMVC配置多视图

    一.背景 1.什么是FreeMarker FreeMarker是一个模板引擎,一个基于模板生成文本输出的通用工具,使用纯Java编写 FreeMarker被设计用来生成HTML Web页面,特别是基于 ...

  2. springmvc 配置多视图(jsp,freemarker,HTML等)

    SpringMVC 的 Controller 可以返回各种各样的视图.比如 JSP, JSON, Velocity, FreeMarker, XML, PDF, Excel, 还有Html字符流 等等 ...

  3. SpringMvc配置自定义视图

    1.在dispatcherServlet-servlet.xml配置自定义视图 <!-- 配置视图 BeanNameViewResolver 解析器: 使用视图的名字来解析视图 --> & ...

  4. springmvc配置多视图 - tiles, velocity, freeMarker, jsp

    转自: http://www.cnblogs.com/shanheyongmu/p/5684595.html <!-- Velocity --> <bean id="vel ...

  5. springmvc 配置多视图,返回jsp,velocity,freeMarker,tiles(模板)等等

    springmvc-servlet.xml配置 <!-- Velocity --> <bean id="velocityViewResolver" class = ...

  6. SpringMVC源码分析6:SpringMVC的视图解析原理

    title: SpringMVC源码分析6:SpringMVC的视图解析原理 date: 2018-06-07 11:03:19 tags: - SpringMVC categories: - 后端 ...

  7. springMVC+ freemark多视图配置

    <!--通用视图解析器--> <bean id="viewResolverCommon" class="org.springframework.web. ...

  8. SpringMVC 返回 html 视图页面,SpringMVC与Servlet,Servlet重定向与转发

    1. SpringMVC与Servlet的关系 SpringMVC框架是建立在Servlet之上的,提供各种功能,各种封装,各种方便的同时,它一点儿也没有限制Servlet,我们完全可以在Spring ...

  9. SpringMVC配置版到注解版

    什么是springmvc? 1.1.什么是MVC MVC是模型(Model).视图(View).控制器(Controller)的简写,是一种软件设计规范. 是将业务逻辑.数据.显示分离的方法来组织代码 ...

随机推荐

  1. ES6新特性概述

    http://es6.ruanyifeng.com/#README https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference ...

  2. IP 协议

    在网络层中,使用的是 ip 协议,它规定网络地址的协议. ip 地址分为两个部分: 网络部分:标识子网 主机部分:标识主机 子网掩码 表示子网络特征的一个参数,它规定 网络部分全部为1,主机部分全部为 ...

  3. 精心收集的 48 个 JavaScript 代码片段,仅需 30 秒就可理解!

    原文:https://github.com/Chalarangelo/30-seconds-of-code#anagrams-of-string-with-duplicates 作者:Chalaran ...

  4. 2018-04-27 搭建Python官方文档翻译环境-汉化示例代码

    通过官方i18n流程, 实现文档中的代码段的汉化, 效果如下(4. More Control Flow Tools): 步骤 基于python官方3.6版文档cpython/Doc生成pot文件. 参 ...

  5. 【代码笔记】Web-JavaScript-javascript while循环

    一,效果图. 二,代码. <!DOCTYPE html> <html> <head> <meta charset="utf-8"> ...

  6. Android图片采样缩放

    为什么要对Android中的图片进行采样缩放呢? 是为了更加高效的加载Bitmap.假设通过imageView来显示图片,很多时候ImageView并没有图片的原始尺寸那么大,这时候把整张图片加载进来 ...

  7. QTP入门——玩玩小飞机

    1.什么是QTP? 百度百科中对QTP是这么介绍的: ——”QTP是QuickTest Professional的简称,是一种自动化测试工具.使用QTP的目的是想用它来执行重复的自动化测试,主要是用于 ...

  8. java后台打开浏览器代码

    import java.awt.Desktop; import java.io.IOException; import java.net.URI; import java.net.URISyntaxE ...

  9. kmp算法python实现

    kmp算法python实现 kmp算法 kmp算法用于字符串的模式匹配,也就是找到模式字符串在目标字符串的第一次出现的位置比如abababc那么bab在其位置1处,bc在其位置5处我们首先想到的最简单 ...

  10. IPerf——网络测试工具介绍与源码解析(2)

    对于IPerf源码解析,我是基于2.0.5版本在Windows下执行的情况进行分析的,提倡开始先通过对源码的简单修改使其能够在本地编译器运行起来,这样可以打印输出一些中间信息,对于理解源码的逻辑,程序 ...