[TOC]
# 1. 数据死循环
当两个实体类相互引用时,这时数据传递到前端会形成死循环问题。
```java
public class Student implements Serializable {
//Student中引用Score
@OneToMany(mappedBy = "student")
private List<Score> scoreList;
}
public class Score implements Serializable {
//Score中引用Student
@ManyToOne(targetEntity = Student.class)
@JoinColumn(name = "student_id", referencedColumnName = "id")
private Student student;
}
```
:-: ![](https://img.kancloud.cn/03/84/0384ed022ecbf8cdc4c495051434b17b_2502x388.png)
死循环结果
后端报如下异常:
```java
java.lang.IllegalStateException: Cannot call sendError() after the response has been committed
at org.apache.catalina.connector.ResponseFacade.checkCommitted(ResponseFacade.java:504) ~[tomcat-embed-core-10.1.8.jar:10.1.8]
```
<br/>
<mark>解决办法:标记注解 @JsonIgnoreProperties 来忽略属性。</mark>
```java
public class Student implements Serializable {
//忽视Score中的student变量
@JsonIgnoreProperties("student")
@OneToMany(mappedBy = "student")
private List<Score> scoreList;
}
public class Score implements Serializable {
//忽视Student里面的scoreList变量
@JsonIgnoreProperties("scoreList")
@ManyToOne(targetEntity = Student.class)
@JoinColumn(name = "student_id", referencedColumnName = "id")
private Student student;
}
```
<br/>
# 2. lombok导致的死循环
lombok 框架中提供的注解`@Data`、`@ToString`、`@EqualsAndHashCode`会用到当前类的所有变量来重写对应的方法,在调用这些方法时就会进入死循环。
```java
public class Student implements Serializable {
@OneToMany(mappedBy = "student")
private List<Score> scoreList;
//Student的toString方法用到了this.getScoreList()
public String toString() {
Integer var10000 = this.getId();
return "Student(id=" + var10000 + ", name=" + this.getName() + ", age=" + this.getAge() + ", scoreList=" + this.getScoreList() + ")";
}
}
public class Score implements Serializable {
@ManyToOne(targetEntity = Student.class)
@JoinColumn(name = "student_id",referencedColumnName = "id")
private Student student;
//Score的toString方法用到了this.getStudent()
public String toString() {
Integer var10000 = this.getId();
return "Score(id=" + var10000 + ", name=" + this.getName() + ", score=" + this.getScore() + ", student=" + this.getStudent() + ")";
}
}
```
两个类的 toString() 方法,都进行了相互引用,这时如果调用 toString() 就会进入死循环并抛出如下异常。
```java
java.lang.StackOverflowError: null
at java.base/jdk.internal.math.FloatingDecimal.getBinaryToASCIIConverter(FloatingDecimal.java:1750) ~[na:na]
at java.base/jdk.internal.math.FloatingDecimal.getBinaryToASCIIConverter(FloatingDecimal.java:1738) ~[na:na]
at java.base/jdk.internal.math.FloatingDecimal.toJavaFormatString(FloatingDecimal.java:70) ~[na:na]
at java.base/java.lang.Double.toString(Double.java:312) ~[na:na]
at java.base/java.lang.Double.toString(Double.java:769) ~[na:na]
at java.base/java.lang.StringConcatHelper.stringOf(StringConcatHelper.java:453) ~[na:na]
at learn.bootjpa.entity.Score.toString(Score.java:9) ~[classes/:na]
```
<br/>
<mark>解决办法1:不调用toString、equals、hashCode方法。</mark>
<mark>解决办法2:重写 toString、equals、hashCode方法,避免相互引用。</mark>
```java
public class Student implements Serializable {
@JsonIgnoreProperties("student")
@OneToMany(mappedBy = "student")
private List<Score> scoreList;
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Student student = (Student) o;
return id.equals(student.id);
}
@Override
public int hashCode() {
return Objects.hash(id);
}
@Override
public String toString() {
List<Score> scoreList2 = new ArrayList<>();
for (Score score : this.getScoreList()) {
Score score2 = new Score();
BeanUtils.copyProperties(score, score2);
score2.setStudent(null);
scoreList2.add(score2);
}
return "Student(id=" + id + ", name=" + name + ", age=" + age + ", scoreList=" + scoreList2 + ")";
}
}
```
- Spring
- Spring是什么
- Spring与EJB对比
- Spring的组成
- 首个Spring程序
- IoC控制反转
- 什么是IoC
- IoC编程
- 依赖注入方式
- 不同变量注入
- AOP面向切面编程
- AOP思想
- AOP实现原理
- AOP关键术语
- AOP编程
- 5种增强方式
- 切入点规则
- 自动装配
- Spring注解开发
- Bean注解
- AOP注解
- 完全注解
- 配置文件拆分
- SpringBean
- Bean常用属性
- Bean作用域
- Bean生命周期
- SpringBoot
- SpringBoot是什么
- 项目创建
- 配置文件
- 配置类型
- 读取配置
- 占位符
- 多环境配置
- 配置优先级
- 更改配置文件
- 自定义IoC容器
- 常用组件
- ApplicationContextAware
- CommandLineRunner
- Boot[Web]
- 引入模板引擎
- 静态资源访问
- 指定首页
- JSP支持
- 注册拦截器
- 注册Servlet组件
- 注册Servlet
- 注册过滤器
- 注册监听器
- 拦截器与过滤器区别
- 文件上传
- 文件下载
- 变更服务器
- Controller层封装
- HttpServletRequest
- 获取请求行
- 获取请求头
- 获取请求体
- Boot[自动配置]
- 自动配置是什么
- 自动配置报告
- 关闭自动配置
- 条件注解
- Boot[场景启动器]
- 场景启动器是什么
- 自定义场景启动器
- Boot[日志]
- 日志框架
- 日志级别
- 日志配置
- 配置文件
- 切换日志
- Boot[邮件任务]
- Boot[定时任务]
- cron表达式
- 起步
- 任务并行
- 注解Scheduled参数
- Boot[异步任务]
- 起步
- 注意事项与原理
- 自定义线程池
- Boot[缓存]
- JSR107缓存技术
- Spring缓存抽象
- 缓存注解
- SpEL表达式
- 起步
- 自定义key生成器
- 工作原理
- Boot[Redis]
- 起步
- 序列化机制
- Boot[Jdbc]
- 起步
- 两个模板类
- JdbcTemplate
- 增删改
- 查询
- NamedParameterJdbcTemplate
- 增删改
- 查询
- 自定义JdbcTemplate
- Boot[JPA]
- SpringDataJPA是什么
- 与JPA、Hibernate的关系
- 起步
- SpringDataJPA原理
- 查询方式
- 方法命名规则查询
- 限制查询结果查询
- 注解Query查询
- 命名参数查询
- SpEL表达式查询
- 原生查询
- 更新与删除
- 查询指定字段
- Specification动态查询
- 分页查询与排序
- 多表查询
- 一对一查询
- 一对多查询
- 多对多查询
- Specification查询
- Query注解查询
- 主键策略
- 单独主键
- 联合主键
- 级联操作
- 加载规则
- 审计功能
- 常用注解
- 避坑指南
- Boot[JSR303]
- JSR303是什么
- 常用约束
- 起步
- 简单校验
- 嵌套校验
- 分组校验
- 自定义约束注解
- 自定义校验工具
- Spring事务
- 事务的作用
- 起步
- 事务参数
- SpringDoc文档
- SpringDoc是什么
- 起步
- 自定义配置
- 常用Doc注解
- JSR303文档
- knife4j文档
- 常用配置
- Boot[RabbitMQ]
- 起步
- Fanout交换机类型
- Direct交换机类型
- Topic交换机类型
- 延迟队列插件
- RabbitListener监听方法
- JWT认证
- 认证流程
- 起步
- 密码加密
- JWT认证实现