如何更改Servlet 3.0 Spring MVC多部分上传表单的字符编码



我有一个非常简单的JSP/Servlet 3.0/Spring MVC 3.1应用程序。

在我的一个页面上,我有多个表单。这些表单中的一个允许用户上传文件,因此它被配置为具有enctype="multipart/form-data"。我在web.xml文件中使用multipart-config元素配置了多部分上传,该元素自Servlet 3.0以来就可用,并在我的spring配置中与<bean id="multipartResolver" class="org.springframework.web.multipart.support.StandardServletMultipartResolver"/>结合使用。

我还配置了Spring的org.springframework.web.filter.CharacterEncodingFilter

我遇到的问题是,我找不到将StandardServletMultipartResolver的默认编码设置为UTF-8的方法,这通常会导致多部分表单中文本字段的内容全部混乱。

有什么办法解决这个问题吗?

提前谢谢。

web.xml配置:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
                  http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
version="3.0">
<display-name>foo-web</display-name>
<context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>
        WEB-INFapplicationContext.xml
    </param-value>
</context-param>
<listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<servlet>
    <servlet-name>foo</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <load-on-startup>0</load-on-startup>
    <multipart-config>
        <max-file-size>52428800</max-file-size>
        <file-size-threshold>5242880</file-size-threshold>
    </multipart-config>
</servlet>
<servlet-mapping>
    <servlet-name>foo</servlet-name>
    <url-pattern>/</url-pattern>
</servlet-mapping>
<filter>
    <filter-name>CharacterEncodingFilter</filter-name>
    <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
    <init-param>
        <param-name>encoding</param-name>
        <param-value>UTF-8</param-value>
    </init-param>
</filter>
<filter-mapping>
    <filter-name>CharacterEncodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>
<filter>
    <filter-name>springSecurityFilterChain</filter-name>
    <filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
</filter>
<filter-mapping>
    <filter-name>springSecurityFilterChain</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>
<welcome-file-list>
    <welcome-file>login</welcome-file>
</welcome-file-list>

由于我找不到使用StandardMultipartResolver设置默认编码的方法,所以我放弃了servlet 3.0配置,转而使用旧的CommonsMultipartResolver

我在springservlet上下文中这样配置它:

<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
    <!-- one of the properties available; the maximum file size in bytes -->
    <property name="maxUploadSize" value="157286400" />
    <property name="maxInMemorySize" value="5242880"/>
    <property name="defaultEncoding" value="utf-8"/>
</bean>

最后,并没有太大的区别,因为在StandardMultipartResolver的框架下,它只是委托给CommonsMultipartResolver

实际上,我发现servlet 3.0方法更麻烦,因为它需要在web.xml和servlet上下文中进行配置,并且您失去了设置默认编码的能力。

在使用Servlet3 API时,我也遇到了编码问题。经过一番研究,我发现Tomcat7中存在一个错误,在某些情况下,无法使用正确的编码读取参数。有一项工作要做。首先,您需要判断它实际上是哪种编码(如果不是默认的iso-8859-1):

request.setCharacterEncoding("UTF-8");

这基本上就是Spring中的CharacterEncodingFilter所做的。到目前为止没有什么奇怪的。现在是诀窍。称之为:

request.getParameterNames()

确保在getParts()之前调用此方法。如果您使用的是Spring,我想您需要在Spring中结束请求之前,在过滤器中执行此操作。调用方法的顺序至关重要。

更新:Tomcat错误已在7.0.41以后的版本中修复,因此如果您使用最新版本的Tomcat,则只需要设置字符编码即可获得正确的结果。

我按照holmis83的建议创建了自己的多部分过滤器,并且在上运行良好

    public class MyMultiPartFilter extends MultipartFilter {
    Logger logger = LoggerFactory.getLogger(MyMultiPartFilter.class);
    @Override
    protected void doFilterInternal(HttpServletRequest request,
            HttpServletResponse response, FilterChain filterChain)
            throws ServletException, IOException {
        request.setCharacterEncoding("UTF-8");
        request.getParameterNames();
        super.doFilterInternal(request, response, filterChain);
    }
}

最新更新