首页 > 代码库 > sitemesh2.x+velocity+springmvc乱码解决方案

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

<script type="text/javascript" src="file:///C:\Program Files (x86)\WizNote\plugins\Notes.Markdown\markdown\marked.min.js"></script><script type="text/javascript" src="file:///C:\Program Files (x86)\WizNote\plugins\Notes.Markdown\markdown\jquery.min.js"></script>

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

<style></style><style></style><script type="text/javascript" src="http://cdn.mathjax.org/mathjax/latest/extensions/MathMenu.js?rev=2.4-beta-2"></script><script type="text/javascript" src="http://cdn.mathjax.org/mathjax/latest/extensions/MathZoom.js?rev=2.4-beta-2"></script><style></style><style></style><style></style>

引言

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

正文

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

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

 

然后再在spring配置velocity试图解析器的时候加入如下代码,其中<property name="contentType" value="http://www.mamicode.com/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/cla17 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这个方法改写成如下

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