spring mvc velocity多视图

时间:2022-10-20 14:14:26

1、ViewResolver
UrlBasedViewResolver 这个东西是根据url 进行路由的。网上搜了

1、order 排序,同名出现各种问题

2、XmlViewResolver,BeanNameViewResolver,ResourceBundleViewResolver  这个

根据配置文件去找不同的view 乱码。。。莫名,而且配置的起来比较麻烦,好处么,就是一个配置文件基本搞定所有页面位置

乱码据说WebApplicationContext 中可以设置某弄过

3、自己写个ViewResolver ,这个比较推荐,也比较靠谱,网上找了2个,觉得还是集成AbstractCachingViewResolver

比较靠谱,下面是拷贝来的

package com.cnynld.web.tpl;

import java.util.Iterator;
import java.util.Locale;
import java.util.Map; import org.springframework.core.Ordered;
import org.springframework.util.StringUtils;
import org.springframework.web.servlet.View;
import org.springframework.web.servlet.ViewResolver;
import org.springframework.web.servlet.view.AbstractCachingViewResolver; /**
* @author loswing
* spring mvc 多view 兼容的 通过后缀来过滤
* FIXME 也可以通过 集成 ViewResolver 不过 AbstractCachingViewResolver是所有resolver的根更靠谱点
*/
public class SpringMvcExtendViewResolver extends AbstractCachingViewResolver implements Ordered{ private int order = Integer.MIN_VALUE; public int getOrder() {
return order;
} public void setOrder(int order) {
this.order = order;
} private Map<String, ViewResolver> resolvers; public void setResolvers(Map<String, ViewResolver> resolvers) {
this.resolvers = resolvers;
}
private String getViewResolverKey(String fileExtension){
Iterator<String> keyIt = resolvers.keySet().iterator();
while (keyIt.hasNext()) {
String viewResolverKey = (String) keyIt.next();
String[] arr = viewResolverKey.split(",");
for (String subKey : arr) {
if(subKey.equals(fileExtension)) return viewResolverKey;
}
}
return null;
} @Override
protected View loadView(String viewName, Locale locale) throws Exception {
String fileExtension = StringUtils.getFilenameExtension(viewName);
if (fileExtension == null) {
return null;
}
String viewResolverKey = getViewResolverKey(fileExtension);
ViewResolver resolver = resolvers.get(viewResolverKey); return resolver == null ? null : resolver.resolveViewName(viewName,locale); } }
     <!--
把下面配置拷贝到spring-mvc.xml中就可以了。
在需要地方返货 以 vm 为后缀的页面就ok
--> <!-- 自定义多视图解析器,根据请求后缀调用相应的视图解析器 -->
<bean id="LdMultipleViewResolver" class="com.cnynld.web.tpl.SpringMvcExtendViewResolver" p:order="">
<property name="resolvers">
<map>
<entry key="vm">
<bean class="org.springframework.web.servlet.view.velocity.VelocityViewResolver">
<property name="cache" value="true"/>
<property name="prefix" value=""/>
<property name="suffix" value="" />
<property name="contentType" value="text/html;charset=UTF-8" />
</bean>
</entry>
</map>
</property>
</bean> <bean id="velocityConfig" class="org.springframework.web.servlet.view.velocity.VelocityConfigurer">
<property name="resourceLoaderPath" value="classpath:/tpl"/>
<!-- <property name="configLocation" value="/WEB-INF/velocity.properties"/> --> <!-- 用下面这个吧 -->
<property name="velocityProperties">
<props>
<prop key="input.encoding">UTF-</prop>
<prop key="output.encoding">UTF-</prop>
<prop key="contentType">text/html;charset=UTF-</prop> <prop key="file.resource.loader.cache">false</prop>
<prop key="file.resource.loader.modificationCheckInterval"></prop>
<prop key="velocimacro.library.autoreload">true</prop> <!-- <prop key="velocimacro.library">macro.vm</prop> --> <prop key="runtime.log.logsystem.class">org.apache.velocity.runtime.log.SimpleLog4JLogSystem</prop>
<prop key="runtime.log">com.sa</prop>
<prop key="runtime.log.error.stacktrace">true</prop>
<prop key="runtime.log.warn.stacktrace">true</prop>
<prop key="runtime.log.info.stacktrace">false</prop>
<prop key="runtime.log.invalid.reference">true</prop>
</props>
</property>
</bean>

下面记录点为了解决这个问题找的笔记

0、spring mvc 莫名
0.1Controller 的一些

//所有Controller 都靠这个开始一切,
public interface Controller {
ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception;
}
WebContentGenerator Controller的基本属性,设置浏览器相关的解析
AbstractController 所有Controller的根,如果要继承,注意 Override这个 handleRequestInternal,不要去弄handleRequest
返回null 就不会触发视图的解析(各种Resolver)
//spring 的注入的的各种
BaseCommandController //Validator、 属性绑定 等等各种的入口,是conroller 类解析的逻辑所在,其他下面都是业务流
AbstractCommandController 我理解为将request 解析类,可以帮助你解决request到object各种问题
setCommandClass 要转换的对象
setCommandName 命令对象的名称
//form相关的control ,理解为Override 这个后handleRequestInternal,根据个人爱好写了好多if else
AbstractFormController 理解为处理form提交的各个过程
formBackingObject 显示的时候form绑定的数据
referenceData 一些其他显示要用的数据,
//各种FormController 我是都没有用过(bpmn 2.0 看下这个吧)
SimpleFormController,CancellableFormController ,AbstractWizardFormController流程化的 form 提交具体可以其看class的properties
只要request 含名字为:_cancel CancellableFormController的 onCancel就会触发如url?xx=&_cancel=wc
//FIXME 理解上有问题,spring mvc 多视图引擎的中 prefix, suffix  就在这里产生的
AbstractUrlViewController ,UrlFilenameViewController
url匹配按照url最长匹配优先,
/long/long
/long/**/abc
/long/**
/**
如请求为“/long/long” 将匹配第一个“/long/long”,但请求“/long/acd”
则将匹配 “/long/**”,如请求“/long/aa/abc”则匹配“/long/**/abc”,如请求“/abc”则将匹配“/**”
MultiActionController 这个是mvc的核心吧,基本所有Resolver都来自这里 //FIXME 这里的Delegate 不是太明白
public (ModelAndView | Map | String | void) actionName(HttpServletRequest request, HttpServletResponse response,
[,HttpSession session] [,AnyObject]);
、返回值:即模型和视图部分; ModelAndView:模型和视图部分 interface Controller 的标准 Map:只返回模型数据,逻辑视图名会根据RequestToViewNameTranslator实现类来计算,稍候讨论; String:只返回逻辑视图名; void:表示该功能方法直接写出response响应(如果其他返回值类型(如Map)返回null则和void进行相同的处理);
这个就是handleRequest 返回null造成viewresolver 不操作,你可以用response 返回各种莫名的东西,如文件流 、actionName:功能方法名字;由methodNameResolver根据请求信息解析功能方法名,通过反射调用; 、形参列表:顺序固定,“[]”表示可选,我们来看看几个示例吧: //表示到新增页面 public ModelAndView toAdd(HttpServletRequest request, HttpServletResponse response); //表示新增表单提交,在最后可以带着命令对象 public ModelAndView add(HttpServletRequest request, HttpServletResponse response, UserModel user); //列表,但只返回模型数据,视图名会通过RequestToViewNameTranslator实现来计算 public Map list(HttpServletRequest request, HttpServletResponse response); //文件下载,返回值类型为void,表示该功能方法直接写响应 public void fileDownload(HttpServletRequest request, HttpServletResponse response) //第三个参数可以是session public ModelAndView sessionWith(HttpServletRequest request, HttpServletResponse response, HttpSession session); //如果第三个参数是session,那么第四个可以是命令对象,顺序必须是如下顺序 public void sessionAndCommandWith(HttpServletRequest request, HttpServletResponse response, HttpSession session, UserModel user) 、异常处理方法,MultiActionController提供了简单的异常处理,即在请求的功能处理过程中遇到异常会交给异常处理方法进行处理,式如下所示: public ModelAndView anyMeaningfulName(HttpServletRequest request, HttpServletResponse response, ExceptionClass exception) MultiActionController会使用最接近的异常类型来匹配对应的异常处理方法,示例如下所示: //处理PayException public ModelAndView processPayException(HttpServletRequest request, HttpServletResponse response, PayException ex) //处理Exception public ModelAndView processException(HttpServletRequest request, HttpServletResponse response, Exception ex)
.x 配置的一些莫名
prefix, suffix 这个东西 这个在返回的时候有用 视图路径=prefix+逻辑视图名+suffix

1、ViewResolver
UrlBasedViewResolver 这个东西是根据url 进行路由的。

<bean
class="org.springframework.web.servlet.view.UrlBasedViewResolver">
<property name="prefix" value="/WEB-INF/" /> <!--安全考虑把模板放到url 访问不到的地方-->
<property name="suffix" value=".jsp" /> <!--所有后缀为.jsp 用下面这个class来过滤 url-->
<property name="viewClass" value="org.springframework.web.servlet.view.InternalResourceView"/> <!--如果不用这个,进行一次内部处理request是访问不到/WEB-INF/-->
</bean>

1.1 url 视图的分发

InternalResourceViewResolver 是UrlBasedViewResolver的imp 而且会根据Controller类来找对应对象,并且根据请求将
对象啊各种放到HttpServletRequest中然后分发对应的视图上去
eg:url:/hello/test.do

//路由配置

@Controller
@RequestMapping(value="/hello")
public class HelloController {
@RequestMapping(value = "/test1", method = RequestMethod.GET)
public ModelAndView getUser() {
ModelAndView mv = new ModelAndView();
mv.addObject("name", " test1 Hello World!!!");
mv.setViewName("/hello");
}
@RequestMapping(value = "/test2")
public String test3(HttpServletRequest request, ModelMap map) {
map.put("name", "test2 hello world");
return "/hello";
}
}

<!--spring-mvc 配置-->

<bean id="JSPViewResolver"class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="viewClass" value="org.springframework.web.servlet.view.JstlView" /><!--如何解析视图-->
<property name="contentType" value="text/html" /><!--客户端生成的页面类型-->
<property name="prefix" value="/WEB-INF/" /><!--模板放在哪里-->
<property name="suffix" value=".jsp" /><!--模板是啥结尾的-->
<!--多视图的时候有用,效率问题,如果没有InternalResourceViewResolver会返回个null然后停滞,造成其他Resolver不执行,所以order要大,建议用ResourceBundleViewResolver-->
<property name="order" value=""></property>
</bean>

//在/WEB-INF/ 建个文件 hello.jsp
hello ${message}

//访问
http://localhost:8080/hello/test1.do
http://localhost:8080/hello/test2.do

2、XmlViewResolver根据配置文件分发页面,这个是我以前最恨的,每次svn都冲突,搞头昏
但是后来觉得这个实际更好用,代码里模板面嵌入模板路径,最后的结果是更加混乱(不过继续推荐ResourceBundleViewResolver)

    <!--spring-mvc-->
<bean class="org.springframework.web.servlet.view.XmlViewResolver">
<property name="location" value="/WEB-INF/views.xml"/> <!--配置文件在哪里-->
<property name="order" value=""/>
</bean>
//views.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
<bean id="myxml" class="org.springframework.web.servlet.view.InternalResourceView">
<property name="url" value="/hello.jsp"/> <!--可以交给美工自己去弄-->
</bean>
</beans>
//路由,上面路由添加
@RequestMapping("/xml")
public String testXmlViewResolver() {
return "myxml"; //这样模板就交给前台配置,后台人员只要告诉前台基本
}

3、ResourceBundleViewResolver和xml很像,不过他的配置文件必须是放在classpath路径下面
默认为views.properties 如果不使用默认值的话,则可以通过属性baseName或baseNames来指定。baseName只是指定一个基名称,
Spring会在指定的classpath根目录下寻找以指定的baseName开始的属性文件进行View解析,如指定的baseName是base,
那么base.properties、baseabc.properties等等以base开始的属性文件都会被Spring当做ResourceBundleViewResolver解析视图的资源文件。

<bean class="org.springframework.web.servlet.view.ResourceBundleViewResolver">
<property name="basename" value="views"></property> <!--所有以views开头的文件都会被解析,建议一个模块一个views_moduleX.properties-->
<property name="order"value=""></property> <!--如果你还配置了其他Resolver这个是非常必要的-->
</bean>
//views.properties
resourceBundle.(class)=org.springframework.web.servlet.view.InternalResourceView
resourceBundle.url=/hello.jsp
//views_hello.properties
hello.(class)=org.springframework.web.servlet.view.InternalResourceView
hello.url=/hello.jsp
//code
@RequestMapping(value = "/test3")
public String test3(HttpServletRequest request, ModelMap map) {
map.put("name", "test3 hello world");
return "hello";
}

参考:
http://haohaoxuexi.iteye.com/blog/1770554 说明白了view的解析(其他博文也建议看下)
https://github.com/sxyx2008/maven-framework-project/ 很好的spring mvc 的demo
http://www.ibm.com/developerworks/cn/java/j-lo-springview/ 视图和视图解析器的相关概念 这个写的一般不过是中文的
http://blog.csdn.net/chichengit/article/details/12098111 写的比较全的,文中大量ctrl+c了其代码和文字
http://martinwuje.iteye.com/blog/1872952 velocity自己搞

spring mvc velocity多视图的更多相关文章

  1. Spring MVC &plus; Velocity实现国际化配置

    国际化介绍 web开发中,国际化是需要考虑的一个问题,而且这个问题一般是越早敲定越好(不然等到系统大了,翻译是个问题).下面是结合实际项目(Spring MVC+Velocity)对实现国际化的一些总 ...

  2. spring mvc&colon; 资源绑定视图解析器&lpar;不推荐&rpar;

    spring mvc: 资源绑定视图解析器(不推荐) 不适合单控制器多方法访问,有知道的兄弟能否告知. 访问地址: http://localhost:8080/guga2/hello/index 项目 ...

  3. spring mvc&colon;内部资源视图解析器2&lpar;注解实现&rpar;&commat;Controller&sol;&commat;RequestMapping

    spring mvc:内部资源视图解析器2(注解实现)  @Controller/@RequestMapping 访问地址: http://localhost:8080/guga2/hello/goo ...

  4. spring mvc&colon;内部资源视图解析器&lpar;注解实现&rpar;&commat;Controller&sol;&commat;RequestMapping

    spring mvc:内部资源视图解析器(注解实现)@Controller/@RequestMapping 项目访问地址: http://localhost:8080/guga2/hello/prin ...

  5. Spring MVC资源绑定视图解析器

    ResourceBundleViewResolver使用属性文件中定义的视图bean来解析视图名称. 以下示例显示如何使用Spring Web MVC框架中的ResourceBundleViewRes ...

  6. Spring MVC 数据模型与视图

      从控制器获取数据后,会装载数据到数据模型和视图中,然后将视图名称转发到视图解析器中,通过解析器解析后得到最终视图,最后将数据模型渲染到视图中,展示最终的结果给用户. 用ModelAndView来定 ...

  7. &lbrack;Spring MVC&rsqb; - JSP &plus; Freemarker视图解释器整合

    Spring MVC中如果只使用JSP做视图,可以使用下面这段即可解决: <!-- 视图解释类 --> <bean class="org.springframework.w ...

  8. spring mvc &plus; velocity 搭建实例程序maven版本并且使用的是tomcat容器而不是jetty(step by step)

    笔者最近在学习spring mvc 查了很多资料,但用jsp的居多,但项目中需要用velocity,所以说就学习了一下,现将所查资料以及搭建过程陈述如下,供需要的人参考 1.楼主用的是eclipse+ ...

  9. &lbrack;Spring MVC&rsqb; - JSP &plus; Freemarker视图解释器整合&lpar;转&rpar;

    Spring MVC中如果只使用JSP做视图,可以使用下面这段即可解决: <!-- 视图解释类 --> <bean class="org.springframework.w ...

随机推荐

  1. CSRF 防御策略

    在业界目前防御 CSRF 攻击主要有三种策略:验证 HTTP Referer 字段:在请求地址中添加 token 并验证:在 HTTP 头中自定义属性并验证.下面就分别对这三种策略进行详细介绍. 验证 ...

  2. CORS 跨域

    跨域请求一直是网页编程中的一个难题,在过去,绝大多数人都倾向于使用JSONP来解决这一问题.不过现在,我们可以考虑一下W3C中一项新的特性——CORS(Cross-Origin Resource Sh ...

  3. CC1310 笔记

    GPIO控制: #include <ti/drivers/GPIO.h> GPIO_init() 函数会调用 结构体实例 GPIOCC26XX_config,把需要使用到的GPIO放一起, ...

  4. centos6&period;9 改系统语言成中文简体

    1.在root权限下 切换到root下:su root 查看当前语言环境:locale -a  (注意中间有空格) 如果看到 zh_CN.UTF-8(这个是中文简体)说明你的系统支持中文语言 2.编辑 ...

  5. 性能测试学习 第七课 --loadrunner中JavaVuser脚本的编写

    1.环境准备:      LoadRunner11----->对应JDK1.6版本(32位) LoadRunner12----->对应JDK1.7版本(32位) (一).JDK下载安装完成 ...

  6. Docker容器服务发现方案

    一.      目的 在服务在容器中部署时,外部调用服务需要知道服务接口ip及端口号,这样导致部署时需要配置,从而增加部署的困难.本文档主要介绍如何使用ningx反向代理和consul进行自动化服务发 ...

  7. ECharts设置y轴显示

    参考地址:https://www.w3cschool.cn/echarts_tutorial/echarts_tutorial-no3h2cul.html yAxis: { type: 'value' ...

  8. 鸟哥的Linux私房菜——第十九章:例行命令的建立

    视频链接:http://www.bilibili.com/video/av11008859/ 1. 什么是例行性命令 (分为两种,一种是周期性的,一种是突发性的)1.1 Linux 工作排程的种类: ...

  9. python &comma; 顺序迭代合并后的list对象

    有一系列排序序列,想将它们合并后得到一个排序序列并在上面迭代遍历 heapq.merge() 函数可以帮你解决这个问题.比如: >>> import heapq >>&g ...

  10. JWT&plus;ASP&period;NET MVC 时间戳防止重放攻击

     时间戳作用 客户端在向服务端接口进行请求,如果请求信息进行了加密处理,被第三方截取到请求包,可以使用该请求包进行重复请求操作.如果服务端不进行防重放攻击,就会服务器压力增大,而使用时间戳的方式可以解 ...