Debug - Method threw 'java.lang.StackOverflowError' exception

 

回到顶部(go to top)

一、问题背景

报错信息:

Method threw 'java.lang.StackOverflowError' exception. Cannot evaluate com.huatai.nats.model.quant.basic.ComparableSecurityMonitoring.toString()

 

 

 

回到顶部(go to top)

二、该错误的本质

根本上是因为实例之间的互相引用

 

 

回到顶部(go to top)

三、本问题的解决方法

3.1 这里有一个Map<String, ComparableSecurityMonitoring> 储存所有的ComparableSecurityMonitoring. key为headsSecurityId

3.2 ComparableSecurityMonitoring 还有一个属性是List<ComparableSecurityMonitoring>ChildrenList

3.3 这里有一个father ComparableSecurityMonitoring (其需要从3.1的map获取),需要填充其ChildrenList(所有的child也需要从3.1的map获取) 

4.一旦用户错误的将 father 设置为自己的child, 那么在设置的时候,就会出现循环引用 -- 我添加我自己到ChildrenList属性里

 

 

回到顶部(go to top)

四、其他类似问题解决方案

详见:https://blog.csdn.net/Saintmm/article/details/117335502

一、背景
今天在做项目中发现A类与B类的关系是1:n,即A类中有一个List<B> bList的成员变量,B类中有一个A a的成员变量。

当执行A.getBList().add(b)方法时会报错:Method threw 'java.lang.StackOverflowError' exception. Cannot evaluate com.saint.start.gaia.A.toString()

 

 

 

二、原因
StackOverFlow问题,顾名思义是栈溢出的意思,Java虚拟机在实际调用方法时会设置最大栈深度,防止其暴走直接占用满所有内存。

根本上是因为实例之间的互相引用!和Spring中的循环依赖很相似。

因为Debugger会调用对象的toString()方法以显示对象中的所有数据信息。然后,我的所有对象都使用lombok的@Data注解重写了toString()方法。问题就出在这里,这意味着程序会不断地调用A类和B类的toString()方法,永不停止,结果就造成了StackOverFlow的问题。

三、解决方案
在类中重写toString()方法,不打印互相引用的类成员变量,避免这种循环调用的产生。

 

四、实例
// A类
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

import java.util.List;

/**
* @author Saint
*/
@Data
@AllArgsConstructor
@NoArgsConstructor
public class Address {
private String name;
private List<User> userList;

@Override
public String toString() {
return name;
}
}

// B类
/**
* @author Saint
*/
@Data
@AllArgsConstructor
@NoArgsConstructor
class User {
private String name;

private Address address;

@Override
public String toString() {
return name;
}
}

// main方法
public class Main {

public static void main(String[] args) {
Method m = new Method();
Address address = new Address("nanjing", new ArrayList<>());
User ha = new User("ha", address);
address.getUserList().add(ha);
User user = new User("start", address);
address.getUserList().add(user);
}
}

 

posted on   frank_cui  阅读(474)  评论(0编辑  收藏  举报

相关博文:
阅读排行:
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?
历史上的今天:
2021-11-17 FI - [框架] 如何掌握一个金融产品
2020-11-17 Hibernate - hibernate 的优化措施
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

导航

统计

levels of contents
点击右上角即可分享
微信分享提示