Как изменить кодировку символов для форм-факсов Servlet 3.0 Spring MVC?

У меня довольно простое приложение JSP/Servlet 3.0/ Spring MVC 3.1.

На одной из моих страниц у меня есть несколько форм. Одна из этих форм позволяет пользователю загружать файл и, таким образом, настраивается с помощью enctype="multipart/form-data". Я сконфигурировал многостраничную загрузку в файле web.xml с элементом multipart-config, доступным с сервлета 3.0, в сочетании с <bean id="multipartResolver" class="org.springframework.web.multipart.support.StandardServletMultipartResolver"/> в моей конфигурации spring.

У меня также есть spring org.springframework.web.filter.CharacterEncodingFilter.

Проблема заключается в том, что я не могу найти способ установить стандартную кодировку StandardServletMultipartResolver для UTF-8, что часто приводит к искажению содержимого текстовых полей в многочастной форме.

Есть ли способ исправить это?

Спасибо заранее.

web.xml config:

<?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-INF\applicationContext.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>

Ответ 1

Поскольку я не нашел способа установить кодировку по умолчанию с помощью StandardMultipartResolver, я сбросил конфигурацию сервлета 3.0 и пошел на старый добрый CommonsMultipartResolver.

Я сконфигурировал его в моем контексте spring сервлета:

<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.

Я нахожу подход сервлета 3.0 более сложным, так как он требует настройки как в web.xml, так и в контексте сервлета, и вы теряете возможность устанавливать кодировку по умолчанию.

Ответ 2

У меня также была проблема с кодировкой при использовании API Servlet 3. После некоторых исследований я обнаружил, что в Tomcat 7 есть ошибка, которая заставляет параметры не считываться с правильной кодировкой при определенных условиях. Существует обход. Во-первых, вам нужно указать, какая кодировка на самом деле (если это не iso-8859-1):

request.setCharacterEncoding("UTF-8");

Это в основном то, что делает CharacterEncodingFilter в Spring. Пока ничего странного. Теперь трюк. Вызов:

request.getParameterNames()

Убедитесь, что этот метод вызывается до getParts(). Если вы используете Spring, я думаю, вам нужно сделать это в фильтре до того, как запрос завершится в Spring. Порядок, к которому применяются методы, имеет решающее значение.

Обновление: Исправлена ​​ошибка Tomcat в версии 7.0.41, поэтому, если вы используете последнюю версию Tomcat вам нужно только установить кодировку символов, чтобы получить правильный результат.

Ответ 3

Я создал свой собственный многостраничный фильтр, предложенный 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);
    }
}