使用配置:jdk18
tomcat8.5.75 (由于10经常会出一些莫名其妙的问题,所以就让它去我的D盘吃灰了)
IDEA 2021.3.2
spring-framework-5.3.15
问题引出:
当我们使用 SpringMVC 和 thymeleaf 进行请求参数的时候,习惯上会使用form搭配注解实现
例如如下代码:
<form th:action="@{}" method=" "></form>
其中@{}里写路径,而 method 里有两种写法:一种是 post ,另一种是 get ,两种写法均会导致控制台或页面输出乱码问题
由于get方法乱码是由于tomcat导致的,所以有根本解决方法,后续会说
而post方法的乱码是因为我们没有设置相应编码,直接进行获取请求参数,况且,如果设置编码,但是如果有
<load-on-startup>1</load-on-startup>
获取参数在前,依然会导致乱码问题,所以我们需要在获取请求参数之前就要设置编码
解决方案
一、get方法乱码解决方案:
在 *tomcat* 文件夹里找到 *conf* 文件夹,里面的 *server.xml* 文件,点击编辑,找到
在这个位置,添加*URIEncoding="UTF-8"*即可解决
二、post方法乱码解决方案:
正因为存在上述问题,所以我们需要考虑,是否存在比servlet执行时间更早的组件。服务器三大组件:1.监听器 2.过滤器 3.servlet,而监听器是用于监听servletContext的创建和销毁,所以考虑只要当前我们访问的地址满足过滤路径,均会被过滤器进行过滤,所以我们选择在过滤器进行编码,且 SpringMVC 已经提供好,直接使用即可,所以我们在 *web.xml* 进行配置(PS:不管把filter写在哪,都会比 servlet 先进行初始化)
其中必须要写
<init-param>
<param-name>encoding</param-name>
<param-value>UTF-8</param-value>
</init-param>
<init-param>
<param-name>forceResponseEncoding</param-name>
<param-value>true</param-value>
</init-param>
通过源码可以看到
首先说一下为什么找到的是这个源码:最主要的原因是参数里有*FilterChain*(过滤器链),有这个参数百分之八九十是执行过滤的
可以看到:
*encoding*可以为空,即我们需要重新设置,否则是默认编码,同时也是null,所以我们上面给*encodeing*设置为*UTF-8*,让它不为空
且 *forceRequestEncodeing* 和 *forceResponseEncodeing* 定义的时候均为 false
因为我们没有设置过 request.getCharacterEncoding() 所以默认还是null,因此第一个if语句就可以执行,也就是执行 *request.setCharacterEncoding(encoding);* 这条语句 (请求编码) 即
<init-param>
<param-name>encoding</param-name>
<param-value>UTF-8</param-value>
</init-param>
如果想要设置响应编码 就需要将下一个 if 语句也执行 , 执行条件可以看到是让 *isForceResponseEncoding()* 为 *true* 所以有
<init-param>
<param-name>forceResponseEncoding</param-name>
<param-value>true</param-value>
</init-param>