Spring Boot 拦截器返回中文乱码的解决方案(附全局优化思路)

在 Spring Boot 项目开发中,我们常常会在 拦截器 中进行登录校验,并在未登录时直接返回 JSON 提示信息。但很多同学会遇到一个问题:中文提示信息出现乱码 (如 ??? 或乱码字符)。


一、问题原因

根源在于:
HTTP 响应的字符编码(Charset)未正确设置,导致前端解析中文失败。

  • 拦截器中直接使用 response.getWriter().write(...) 输出字符串;
  • Tomcat 默认使用 ISO-8859-1 编码,不支持中文;
  • 前端一般默认用 UTF-8 解析,编码不一致 → 乱码。

二、临时方案:手动设置响应编码

这是最快速见效的办法。在 response.getWriter().write(...) 之前,明确设置字符编码和 Content-Type:

java 复制代码
response.setCharacterEncoding("UTF-8");
response.setContentType("application/json;charset=UTF-8");
response.getWriter().write("{\"code\":401,\"msg\":\"请先登录!\"}");

修改拦截器示例:

java 复制代码
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    String token = request.getHeader(jwtProperties.getTokenName());
    if (StringUtils.isBlank(token)) {
        response.setCharacterEncoding("UTF-8");
        response.setContentType("application/json;charset=UTF-8");
        response.getWriter().write("{\"code\":401,\"msg\":\"请先登录!\"}");
        return false;
    }
    return true;
}

👉 这样可以立刻解决乱码问题,但存在硬编码 JSON、不可复用的问题。


三、优雅方案:使用统一的 Result

如果项目里已经有统一的返回对象(如 Result),建议复用,避免硬编码 JSON。

1. 注入 ObjectMapper

Spring Boot 默认集成了 Jackson,可直接注入:

java 复制代码
@Autowired
private ObjectMapper objectMapper;

2. 生成统一响应

Result 封装后,再转成 JSON 输出:

java 复制代码
Result<?> result = Result.fail(401, "请先登录!");
response.setCharacterEncoding("UTF-8");
response.setContentType("application/json;charset=UTF-8");
response.getWriter().write(objectMapper.writeValueAsString(result));

这样既能保证编码正确,又能保持返回格式一致。


四、根治方案:全局配置响应编码

如果你不想每次都手动设置编码,可以通过全局配置彻底解决。

方式 1:在 application.yml 配置

yaml 复制代码
spring:
  http:
    encoding:
      force: true
      charset: UTF-8
      enabled: true

方式 2:全局配置消息转换器

WebMvcConfig 中统一设置编码:

java 复制代码
@Bean
public StringHttpMessageConverter stringHttpMessageConverter() {
    StringHttpMessageConverter converter = new StringHttpMessageConverter(StandardCharsets.UTF_8);
    return converter;
}

这样项目中所有返回字符串的响应都会默认使用 UTF-8。


五、验证效果

  1. 启动项目后,访问需要登录的接口(不带 Token);
  2. 查看返回结果:
json 复制代码
{"code":401,"msg":"请先登录!"}

若能正常显示中文而不是乱码,说明问题已解决 ✅。

相关推荐
青云交8 小时前
Java 大视界 -- 基于 Java 的大数据实时流处理在工业物联网设备故障预测与智能运维中的应用
java·flink·kafka·工业物联网·设备故障预测·智能运维·实时流处理
codervibe8 小时前
闲鱼商品搜索爬虫:从签名算法到反爬机制的完整逆向与实现
后端
fat house cat_8 小时前
为什么RocketMQ选择mmap+write?RocketMQ零拷贝技术深度解析
java·rocketmq·零拷贝
跟着珅聪学java9 小时前
vue通过spring boot 下载文件教程
前端·spring boot·后端
云闲不收9 小时前
golang编译
开发语言·后端·golang
月疯9 小时前
FLASK与JAVA的文件互传并带参数以及流上传(单文件互传亲测)
java·python·flask
数据知道9 小时前
Go语言:加密与解密详解
开发语言·后端·golang·go语言
XUE-52113149 小时前
BGP实验-路由优选
linux·服务器·网络·网络协议
武子康9 小时前
大数据-117 - Flink JDBC Sink 详细解析:MySQL 实时写入、批处理优化与最佳实践 写出Kafka
大数据·后端·flink
Stream_Silver9 小时前
LangChain入门实践3:PromptTemplate提示词模板详解
java·python·学习·langchain·language model