Locale+Jackson导致Controller接口StackOverflowError异常解决

news2025/2/27 20:23:05

问题

由于参与的项目有出海需求,即需要给外国人使用,即:需要支持i18n(Internationalization的缩写,共20个字母,除去首尾两个字母,中间有18个,故简称i18n)。

本来是好的,非常简单的Controller接口,在增加字段后,突然爆出StackOverflowError异常:

org.springframework.web.util.NestedServletException: Handler dispatch failed; nested exception is java.lang.StackOverflowError
	at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:1087)
	at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:965)
	at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1006)
	at org.springframework.web.servlet.FrameworkServlet.doPost(FrameworkServlet.java:909)
	at javax.servlet.http.HttpServlet.service(HttpServlet.java:555)

排查

直接请教ChatGPT,给出如下几种可能导致Controller层接口StackOverflowError异常的场景:

  • Controller方法导致无限递归:直接或间接调用自己
  • JSON序列化导致无限递归:Controller返回的对象包含双向引用,Jackson序列化时可能会陷入无限递归。
@Data
@Entity
public class User {
	@Id
	private Long id;
	private String name;
	
	@OneToMany(mappedBy = "user")
	private List<Order> orders;
}

@Data
@Entity
public class Order {
	@Id
	private Long id;
	private String product;
	
	@ManyToOne
	private User user; // 双向引用
}

当User关联Order,Order也关联User,Jackson解析时会无限嵌套,导致StackOverflowError。

解决方案
使用@JsonManagedReference和@JsonBackReference解决循环引用:

@Data
@Entity
public class User {
	@Id
	private Long id;
	private String name;
	
	@OneToMany(mappedBy = "user")
	@JsonManagedReference
	private List<Order> orders;
}

@Data
@Entity
public class Order {
	@Id
	private Long id;
	private String product;
	
	@ManyToOne
	@JsonBackReference
	private User user;
}

或使用@JsonIgnore:

@ManyToOne
@JsonIgnore
private User user;
  • AOP拦截器陷入无限调用:错误使用Spring AOP或拦截器,可能会导致方法被无限调用;
  • 错误的toString()方法:实体类toString()方法递归调用自身字段。

四种可能性,第二种可能性最大;于是将问题定位到JSON序列化上。

我的Controller层接口,返回对象(即responseBody)并没有包含双向引用;于是将注意力转移到接口的requestBody上。

定位

经过分析,是我在@RequestBody标注的Dto实体类里新增一个Locale字段:

@EqualsAndHashCode(callSuper = true)
@Data
@AllArgsConstructor
@NoArgsConstructor
@Slf4j
@ApiModel(description = "消息创建请求体")
public class MessageCreateDto {
	private Locale locale;
}

去掉这个字段,就不再有StackOverflowError异常。

方案

定位到问题后,怎么解决问题呢?

还是直接请教ChatGPT,给出的分析:

Locale本身不是一个普通的Java Bean,它没有默认的无参构造方法,并且Jackson可能无法正确解析它,导致JSON反序列化时进入递归调用,最终导致StackOverflowError。

给出的几种解决方案:

方案一:使用@JsonCreator和@JsonValue

import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonValue;

public class MessageCreateDto {
	private Locale locale;
	@JsonCreator
	public static Locale fromString(String value) {
		// 解析 "en_US" => Locale("en", "US")
		return Locale.forLanguageTag(value.replace('_', '-'));
	}
	
	@JsonValue
	public String toJson() {
		// 让 Jackson 以 "en-US" 格式序列化
		return locale.toLanguageTag();
	}
}

经过验证:并没有解决问题。

方案二:自定义Jackson反序列化器

import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonDeserializer;

public class LocaleDeserializer extends JsonDeserializer<Locale> {
	@Override
	public Locale deserialize(JsonParser p, DeserializationContext ctxt) throws IOException {
		String localeStr = p.getText();
		return Locale.forLanguageTag(localeStr.replace('_', '-'));
	}
}

然后在实体类加上注解:

	@JsonDeserialize(using = LocaleDeserializer.class)
	@JsonSerialize(using = LocaleSerializer.class)
	private Locale locale;

经过验证:并没有解决问题。

再加上序列化器,还是有StackOverflowError异常。

import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.SerializerProvider;

public class LocaleSerializer extends JsonSerializer<Locale> {
	@Override
	public void serialize(Locale locale, JsonGenerator gen, SerializerProvider serializers) throws IOException {
		if (locale != null) {
			// 格式化为 "en-US" 这种形式,而不是 "en_US"
			gen.writeString(locale.toLanguageTag());
		} else {
			gen.writeNull();
		}
	}
}

方案三:手动转换String

public class MessageCreateDto {
    private String projectId;
    // 直接使用Locale类型,Jackson反序列化报错StackOverflowError
    private String locale;

    public Locale getLocale() {
        return Locale.forLanguageTag(locale.replace('_', '-'));
    }
}

经过验证,此方案可行。

反思

上面的方案三虽然可以解决问题。

但是!!

遗留问题:如果我有5个Java应用,每个应用都有20~30个不等的Controller层接口需要支持i18n,那我得在每个应用,每个Controller层接口里加上String locale字段,然后再增加一个getLocale方法么?

稍微熟悉HTTP,或有一点点前端开发经验,或使用F12快捷键查看过Chrome控制台,就知道有个accept-language HTTPheader:
在这里插入图片描述
因此,不是加字段,而是:

  • 前端在全局配置文件里设置accept-language
  • 所有请求接口里自动带上此header;
  • 后端接口按需解析HTTP header,然后做i18n处理。

题外话

zh_CN和zh-CN

关于Locale,到底是使用下划线(即,_)还是横杠(即,-,专业说法,其实是减号连字符

参考zh-CN还是zh_CN。

另外Java API也该出它的立场,zh_CN是老式写法,zh-CN才是推荐的写法。如下图所示,有一个replace动作:
在这里插入图片描述

Locale.US.toString()和Locale.US.toLanguageTag()

前者是老式写法(中间有空格),后者是新式写法:
在这里插入图片描述

参考

  • ChatGPT

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2307032.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

安卓工控平板电脑在环境监测设备中的运用

安卓工控平板电脑在环境监测设备中的运用主要体现在以下几个方面&#xff1a; 一、耐用性与可靠性 安卓工控平板电脑通常具有坚固耐用的外壳设计&#xff0c;如铝合金面板和镀锌钢板箱体结构&#xff0c;能够抵抗高温、低温、湿度、震动等恶劣的工作环境。这种耐用性和可靠性…

【洛谷排序算法】P1012拼数-详细讲解

洛谷 P1012 拼数这道题本身并非单纯考察某种经典排序算法&#xff08;如冒泡排序、选择排序、插入排序、快速排序、归并排序等&#xff09;的实现&#xff0c;而是在排序的基础上&#xff0c;自定义了排序的比较规则&#xff0c;属于自定义排序类型的题目。不过它借助了标准库中…

文心一言AI创意画

介绍 文心一言是百度推出的新一代知识增强大语言模型&#xff0c;属于文心大模型家族的新成员。‌它能够与人对话互动、回答问题、协助创作&#xff0c;高效便捷地帮助人们获取信息、知识和灵感。‌ 特点 文心一言基于数万亿数据和数千亿知识进行融合学习&#xff0c;采用预训…

java项目之基于ssm的图书馆书库管理系统(源码+文档)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于ssm的图书馆书库管理系统。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 该系统可以实现图书信息管理…

使用OpenCV实现帧间变化检测:基于轮廓的动态区域标注

在计算机视觉中&#xff0c;帧间差异检测&#xff08;frame differencing&#xff09;是一种常用的技术&#xff0c;用于检测视频流中的动态变化区域。这种方法尤其适用于监控、运动分析、目标追踪等场景。在这篇博客中&#xff0c;我们将通过分析一个基于OpenCV的简单帧间差异…

deepseek从入门到精通-第一篇.本地化部署

前言 自从22年年底开始&#xff0c;人工智能开始从实验室一下子走入了普通人的视野中&#xff0c;chatgpt像一颗石子投入水中&#xff0c;溅起了一波又一波的涟漪。我们都通过各种方式试用大预言模型和机器进行对话或者提问。随着大语言模型的出现&#xff0c;各个类型的大模型…

2025年SCI一区智能优化算法:真菌生长优化算法(Fungal Growth Optimizer,FGO),提供MATLAB代码

一. 真菌生长优化算法&#xff08;FGO&#xff09; 真菌生长优化算法&#xff08;Fungal Growth Optimizer&#xff0c;FGO&#xff09;是一种新型的自然启发式元启发式算法&#xff0c;其灵感来源于自然界中真菌的生长行为。该算法通过模拟真菌的菌丝尖端生长、分支和孢子萌发…

Ubutu部署WordPress

前言 什么是word press WordPress是一种使用PHP语言开发的建站系统&#xff0c;用户可以在支持PHP和MySQL数据库的服务器上架设WordPress。它是一个开源的内容管理系统&#xff08;CMS&#xff09;&#xff0c;允许用户构建动态网站和博客。现在的WordPress已经强大到几乎可以…

BIO、NIO、AIO解析

一、基础概念 1、IO的含义 IO&#xff0c;Input/Output&#xff0c;即输入/输出。从计算机结构来看&#xff0c;IO描述了计算机系统和外部设备之间通讯的过程。从应用程序角度来看&#xff0c;一个进程的地址空间划分为 用户空间&#xff08;User space&#xff09; 和 内核空…

【Python网络爬虫笔记】14-使用代理绕过访问限制

【Python网络爬虫笔记】14-网络代理 目录什么是代理&#xff1f;为什么需要使用代理&#xff1f;代理的类型如何在Python中使用代理&#xff1f;使用requests库设置代理使用urllib库设置代理使用scrapy框架设置代理 典型案例&#xff1a;使用代理爬取豆瓣电影Top250步骤1&#…

Linux中Shell运行原理和权限(下)(4)

文章目录 前言一、Shell的运行原理二、Linux当中的权限问题Linux权限的概念如何将普通用户添加到信任列表 三、Linux权限管理文件访问者的分类&#xff08;人&#xff09;文件类型和访问权限&#xff08;事物属性&#xff09;文件权限值的表示方法文件访问权限的相关设置方法如…

OceanBase数据库实战:Windows Docker部署与DBeaver无缝对接

一、前言 OceanBase 是一款高性能、高可扩展的分布式数据库&#xff0c;适用于大规模数据处理和企业级应用。 随着大数据和云计算的普及&#xff0c;OceanBase 在企业数字化转型中扮演着重要角色。学习 OceanBase 可以帮助开发者掌握先进的分布式数据库技术&#xff0c;提升数…

技术速递|.NET 9 网络优化

作者&#xff1a;Mňa&#xff0c;Natalia&#xff0c;Anton 排版&#xff1a;Alan Wang 秉承我们的传统&#xff0c;我们很高兴与您分享这篇博客文章&#xff0c;以介绍新的 .NET 版本中网络领域相关的最新动态和最有趣的变化。今年&#xff0c;我们带来了 HTTP 领域的改变、新…

Tag标签的使用

一个非常适合运用在vue项目中的组件&#xff1a;Tag标签。 目录 一、准备工作 1、安装element-plus库 2、配置element-plus库 二、Tag标签入门 1、打开element官网&#xff0c;搜索tag标签 2、体验Tag标签的基础用法 三、Tag标签进阶训练1 1、定义一个数组&#xff0c;…

Linux:(3)

一&#xff1a;Linux和Linux互传&#xff08;压缩包&#xff09; scp:Linux scp 命令用于 Linux 之间复制文件和目录。 scp 是 secure copy 的缩写, scp 是 linux 系统下基于 ssh 登陆进行安全的远程文件拷贝命令。 scp 是加密的&#xff0c;rcp 是不加密的&#xff0c;scp 是…

HarmonyOS 5.0应用开发——鸿蒙接入高德地图实现POI搜索

【高心星出品】 文章目录 鸿蒙接入高德地图实现POI搜索运行结果&#xff1a;准备地图编写ArkUI布局来加载HTML地图 鸿蒙接入高德地图实现POI搜索 在当今数字化时代&#xff0c;地图应用已成为移动设备中不可或缺的一部分。随着鸿蒙系统的日益普及&#xff0c;如何在鸿蒙应用中…

计算机视觉(opencv-python)入门之常见图像处理基本操作(待补充)

图像预处理是计算机视觉任务中的关键步骤&#xff0c;它通过对原始图像进行处理&#xff0c;以提高后续图像分析、特征提取和识别的准确性。 示例图片 目录 常见图像预处理方法 灰度化处理 法一 法二 说明 切片截取部分图像数据 cv2.cvtColor() 颜色空间转换 cv2.spli…

采用DDNS-GO与cloudflare实现双域名同时访问NAS

这个标题其实解释的还不够清楚&#xff0c;本人是小白&#xff0c;但是买了群晖的NAS后自己瞎折腾了一下&#xff0c;遇到了如下的问题&#xff1a; 1、家里是移动宽带&#xff0c;没有公网IP&#xff0c;因此Ipv4无法使用&#xff0c;IPV6可以正常使用。 2、办公室场地采用的…

w803|联盛德|WM IoT SDK2.X测试|pinout|(2):w803开发板简介

概述 W803-Pico是一款基于联盛德W803芯片为主控的开发板&#xff0c;支持IEEE802.11 b/g/n Wi-Fi&#xff0c;以及BT/BLE4.2协议蓝牙。芯片内置高性能32位处理器&#xff0c;主频高达240MHz。内置2MB Flash以及288KB RAM。硬件采用DIP封装&#xff0c;PCB板载天线&#xff0c;…

【UCB CS 61B SP24】Lecture 16 - Data Structures 2: ADTs, BSTs学习笔记

本文首先介绍了抽象数据类型与树的概念&#xff0c;接着重点讲解二叉搜索树的定义与操作方式&#xff0c;并用 Java 实现一个标准的二叉搜索树结构。 1. 抽象数据类型 首先引入一个概念叫做抽象数据类型&#xff08;Abstract Data Type&#xff0c;ADT&#xff09;&#xff0…