使用 Spring MVC Web 应用程序强制以 html 形式进行 UTF8 编码

Force UTF8 encoding in html form with Spring MVC Web Application

我有一个 spring MVC 网络应用程序并使用 JSP 创建我的网页。 问题是,当我直接提交表单时,编码是 "application/x-www-form-urlencoded"。当我通过 AJAX-Request 从同一表单提交相同数据时,编码为 "application/x-www-form-urlencoded; charset=UTF-8"。

我需要输入 utf8 编码的字符作为我控制器中的用户。例如:用户键入“äöüß”我的控制器得到“äã¶Ã¼Ã”。当我通过 AJAX-Request 发送数据时,我得到正确的“äöüß”。

我做错了什么?这是通过 http-post 提交的简单表单。这个utf8编码也不是不可能。

我的应用程序 运行 在 tomcat 8.5.11 和 Spring 5.0.1 上运行。网页都在 HTML5 中,我在 servlet 3.1 环境中使用 JSTL 1.2.5。 JSON 映射和序列化由 fasterxml 2.9.2

完成

配置完全Java-based.

我的 WebAppInitializer(a.k.a。web.xml)

...
@Override
protected Filter[] getServletFilters() {
    return new Filter[] { new HiddenHttpMethodFilter(), 
                            new CharacterEncodingFilter("UTF-8", true, true) };
}

在我的 servlet 配置中,我明确地为 StringHttpMessageConverter 设置了一个字符集。

...
@Override
public void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
    converters.add(new StringHttpMessageConverter(Charset.forName("UTF-8")));
    converters.add(new ResourceHttpMessageConverter());
    converters.add(new MappingJackson2HttpMessageConverter());
}

网页看起来像

<%@page contentType="text/html;charset=UTF-8"%>
<%@page pageEncoding="UTF-8"%>
<%@page session="true" %>
<%@page trimDirectiveWhitespaces="true" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="fmt" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn" %>
<%@ taglib prefix="sec" uri="http://www.springframework.org/security/tags" %>
<!DOCTYPE html>

<html lang="de">

...

<meta charset="UTF-8"/>

...

<form id="createArticleForm" action="<c:url value='/article/save' />" method="post">
  <input type="hidden" name="utf8" value="&#x2713;" />

...

  <input type="text" name="name" required="required" />

...
</form>

如您所见,我还尝试了带有隐藏字段的 utf8 hack。但没有任何效果。即使我设置了表单属性 accept-charset="UTF-8" and/or encoding="application/x-www-form-urlencoded; charset=UTF-8" 也没有任何变化。

编辑1 我检查了从浏览器发送到服务器的 HTTP 请求 Header。我发现所有参数都是正确的。所以我假设 spring 配置问题。

我自己找到了解决方案。问题出在characterEncodingFilter的初始化上。

在 web.xml 中,您执行以下操作:

<filter>
    <filter-name>encodingFilter</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>
    <init-param>
        <param-name>forceEncoding</param-name>
        <param-value>true</param-value>
    </init-param>
</filter>

<filter-mapping>
    <filter-name>encodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

由于 Spring 5 Java-based Annotation Configuration 受到青睐,根据文档,您可以在 WebApplicationInitializer 中执行以下操作,这比上面的代码简单得多:

@Override
protected Filter[] getServletFilters() {
    return new Filter[] { new HiddenHttpMethodFilter(), 
                            new CharacterEncodingFilter("UTF-8", true, true) };
}

但是这是对每个传入的 HttpRequest 都能正常工作!!!

解决方案是不使用提供的便捷方式自动将过滤器映射到 servlet。

相反,您必须:

@Override
public void onStartup(ServletContext servletContext) throws ServletException {

      FilterRegistration.Dynamic filterRegistration = servletContext.addFilter("characterEncodingFilter", new CharacterEncodingFilter("UTF-8", true, true));
      filterRegistration.addMappingForUrlPatterns(null, false, "/*");

      filterRegistration = servletContext.addFilter("hiddenHttpMethodFilter", new HiddenHttpMethodFilter() );
      filterRegistration.addMappingForUrlPatterns(null, false, "/*");

    super.onStartup(servletContext);
}

在这里您可以看到过滤器映射,这非常完美!