sitemesh2.x+velocity+springmvc乱码解决方案

sitemesh2.x+velocity乱码解决方案.md

引言

通常我们在采用springmvc+velocity架构的时候只需要跳转到action然后在转回html页面,此时即可通过velocity的固有语法在html中取出各种变量。当当我们想在以上的架构中加入sitemesh2.x 的时候会发现配置装饰页面时采用action会出现一些错误,而只能直接使用.vm来配置装饰器页面,可是这样直接跳转的做法很多时候会出现一些乱码问题。

正文

有人会说,可以在velocity的配置文件中加入字符集设置,配置如下:

1 input.encoding = UTF-8
2  output.encoding = UTF-8
3 response.setContentType("text/html;charset=utf-8");    
4 request.setContentType("text/html;charset=utf-8");

 

然后再在spring配置velocity试图解析器的时候加入如下代码,其中<property name="contentType" value="text/html;charset=utf-8" />约定了编码格式为utf-8

 1 <!-- Velocity视图解析器 -->
 2     <bean id="viewResolver"
 3         class="org.springframework.web.servlet.view.velocity.VelocityViewResolver">
 4             <property name="prefix" value="" />
 5             <property name="suffix" value=".html" />
 6             <!-- Whether we should cache views, once resolved -->
 7             <property name="cache" value="false" />
 8             <property name="contentType" value="text/html;charset=utf-8" />
 9             <!-- 暴露Spring本身的宏 -->
10             <property name="exposeSpringMacroHelpers" value="true" />
11 
12             <!-- 格式化日期 :$dateTool.format("yyyy-MM-dd",$!{})  -->
13             <property name="dateToolAttribute" value="dateTool" />
14             <!-- 格式化数字 :  -->
15             <property name="numberToolAttribute" value="numberTool" />
16             <property name="toolboxConfigLocation" value="/WEB-INF/cla
17 
18 sses/velocity/toolbox.xml" />
19     </bean>

 

我相信以上方法已经解决了部分人的乱码问题,但是这一类相关文章我也看了很多,我的问题依然没有解决。
其实上述方法无法解决问题的时候,还有一套终极手段,就是修改服务器的编码方式为utf-8,这种方案通常可以解决绝大部分的乱码问题。但是,这种方法不到万不得已的时候不能使用,因为很有可能公用服务器的时候,你根本没有权限甚至你根本就不能去修改服务器的编码方式,因为这很可能对公用服务器造成一系列未知问题。
我出现的乱码情况比较奇异,因为我访问的内容页是 localhost:8080/sourceDemo/index.action ,而按照sitemesh的组合后的页面的规则来看实际应该就是 访问test.vm这个文件,在这个文件中我有如下变量声明:

其中${body} 是取得用户访问的页面的body标签内的所有元素,而我得到的结果是
哈利路亚四个字乱码了,也就是说装饰器页面乱码,而用户访问的页面没有乱码。
然后通过velocity解析成html页面
于是乎,出于上述考虑,我便决定查看sitemesh源码。
以下先给出我的sitemesh在配置文件中的相应配置:
web.xml中的配置:

 1 <!-- sitemesh -->
 2     <filter>
 3             <filter-name>sitemesh</filter-name>
 4             <filter-class>com.opensymphony.module.sitemesh.filter.PageFilter</filter-class>
 5     </filter>
 6     <filter-mapping>
 7             <filter-name>sitemesh</filter-name>
 8             <url-pattern>/*</url-pattern>
 9     </filter-mapping>
10     <!-- sitemesh -->
11 
12     <!-- sitemesh servlet配置 START -->
13     <servlet>
14             <servlet-name>site-mesh-velocity</servlet-name>
15             <servlet-class>com.cloudwinker.source.sitemesh.module.velocity.VelocityDecoratorServlet</servlet-class>
16             <load-on-startup>1</load-on-startup>
17     </servlet>
18 
19     <servlet-mapping>
20             <servlet-name>site-mesh-velocity</servlet-name>
21             <url-pattern>*.vm</url-pattern>
22     </servlet-mapping>
23     <!-- sitemesh servlet配置 END -->

 

sitemesh.xml中的配置

<sitemesh>
         <property name="decorators-file" value="/WEB-INF/config/sitemesh/decorators.xml" /> 
        <excludes file="${decorators-file}" /> 

    <page-parsers>
            <parser content-type="text/html" class="com.opensymphony.module.sitemesh.parser.HTMLPageParser" />
        </page-parsers>

        <decorator-mappers>

            <mapper class="com.opensymphony.module.sitemesh.mapper.PageDecoratorMapper">
                <param name="property.1" value="meta.decorator" />
                <param name="property.2" value="decorator" />
            </mapper>

            <mapper class="com.opensymphony.module.sitemesh.mapper.FrameSetDecoratorMapper">
            </mapper>

            <mapper class="com.opensymphony.module.sitemesh.mapper.AgentDecoratorMapper">
                <param name="match.MSIE" value="ie" />
                <param name="match.Mozilla [" value="ns" />
                <param name="match.Opera" value="opera" />
                <param name="match.Lynx" value="lynx" />
            </mapper>

            <mapper class="com.opensymphony.module.sitemesh.mapper.PrintableDecoratorMapper">
                <param name="decorator" value="printable" />
                <param name="parameter.name" value="printable" />
                <param name="parameter.value" value="true" />
            </mapper>

            <mapper class="com.opensymphony.module.sitemesh.mapper.RobotDecoratorMapper">
                <param name="decorator" value="robot" />
            </mapper>

            <mapper class="com.opensymphony.module.sitemesh.mapper.ParameterDecoratorMapper">
                <param name="decorator.parameter" value="decorator" />
                <param name="parameter.name" value="confirm" />
                <param name="parameter.value" value="true" />
            </mapper>

            <mapper class="com.opensymphony.module.sitemesh.mapper.FileDecoratorMapper">
            </mapper>

            <mapper class="com.opensymphony.module.sitemesh.mapper.ConfigDecoratorMapper">
                    <param name="config" value="${decorators-file}" />
            </mapper>

        </decorator-mappers>

</sitemesh>

 

decorator.xml中的配置

<?xml version="1.0" encoding="ISO-8859-1"?>

<decorators defaultdir="/decorators">
    <!-- Any urls that are excluded will never be decorated by Sitemesh -->
<!--     <excludes> -->
<!--         <pattern>/frontpage/pages/source/product/product_para.jsp</pattern> -->
<!--     </excludes> -->

<decorator name="adminMain" page="/test.vm"> 
         <pattern>/views/admin/*</pattern>
</decorator>

<!--      <decorator name="adminFooter" page="views/admin/pages/decorators/foot.html"/>  -->
</decorators>

 

其中根目录下得test.vm即为装饰器页面,我将用它来装饰内容页
sitemesh是通过filter来实现的,因此入口其实就是filter部分,
而我配置在web.xml的filter中的部分,实际上是一个与velocity结合的工具,其原本的配置并非是这个,而是com.opensymphony.module.sitemesh.velocity.VelocityDecoratorServlet

<servlet>
            <servlet-name>site-mesh-velocity</servlet-name>
            <servlet-class>com.cloudwinker.source.sitemesh.module.velocity.VelocityDecoratorServlet</servlet-class>
            <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
            <servlet-name>site-mesh-velocity</servlet-name>
            <url-pattern>*.vm</url-pattern>
    </servlet-mapping>

 

我使用jd-gui工具反编译了sitemesh的源码,发现他实际上继承了org.apache.velocity.tools.view.servlet.VelocityViewServlet;也就是velocity-tools-2.0的jar包,目的就在于将sitemesh出来的页面整合到vm文件中去
这是我修改后的VelocityDecoratorServlet

该文件是直接copysitemeshjar包种的org.apache.velocity.tools.view.servlet.VelocityViewServlet 内容,其中最后一句发现
getTemplate(template)实际上是取得velocity的api中的velocity的一个方法,我从debug中得到最后return velocityTemplate时,该对象的encoding属性为ISO-8859-1 ,因为这个原因导致最终显示在页面时,原本属于test.vm的数据全部显示乱码!因此后续的内容就变更为改变在省城velocity模板之前,改变文件的encoding属性!

于是我通过debug追踪,找到了这个文件,也就是velocity-tool-2.0jar包中的 org.apache.velocity.tools.view.velocityView.java这个文件,我截取部分源码给大家看一下:

这就是最后实际调用的方法,从方法内部发现,当encoding==null时实际上使用的是默认的编码方式。我在debug时发现代码运行的这一句的时候,实际encoding的值是null。
因此无论我们在velocity文件中如何配置,都只能得到ISO-8859-1的编码格式文件,于是中文乱码就成了必然情况。
知道了原因后,剩下的就是修改代码了。之所以这里的encoding是null,我想很有可能跟springmvc+velocity 的配置时,实际上使用的spring的响应机制,因此直接从actiong访问时所有编码正常,而直接sitemesh将vm作为装饰器文件时却出现乱码情况,因为sitemesh只是简单的在web.xml中配置了filter,并没有与spring做相应整合。
于是理论上解决方案有两种:其中一种就是改写sitemesh的filter,使之能够在spring的机制中使用,这样的话就能透过spring来取得velocity的相关编码的配置了。由于我太懒了,目前懒得用这种方式去改写,因为嫌他太麻烦。
第二种方式:实际上就是重写org.apache.velocity.tools.view.velocityView.java 这个文件,将getTemplate这个方法改写成如下

之后重启服务器,运行,问题解决!

 

posted @ 2014-07-05 13:01  Andrew Lee  阅读(1051)  评论(0编辑  收藏  举报