如何解决循环引用的问题

本文已收录于专栏
《Java》

概念说明

  循环引用是指在对象之间存在相互引用的情况。具体来说,当一个对象A引用了另一个对象B,而对象B又引用了对象A,它们之间形成了一个循环引用关系。
  循环引用可能会导致一些问题,特别是在序列化和反序列化过程中。在序列化过程中,如果不进行特殊处理,循环引用会导致无限递归的序列化,最终导致栈溢出异常。在反序列化过程中,循环引用可能会导致对象的重复创建,破坏对象的一致性。

发现问题

  当我们实现了类之间的多对多关系的时候,当我们从数据库中查询相关数据的时候,会发现查询出来的结果会存在对象之间嵌套的问题,最后导致堆栈溢出的问题。
Classes类

@Service
@Entity
@Table(name="Classes")
public class Classes implements ApplicationRunner {

    private Long id;

    private Long classId;

    private String className;

    private Integer isDelete;


    @OneToMany(mappedBy = "classes")

    private List<ClassesActor> actorClasses;

	//getter和setter方法
	
	}

Actor类

@Service
@Entity
@Table(name = "Actor")
public class Actor  {
    private Long id;

    private String code;

    private String name;

    private String phone;

	@OneToMany(mappedBy = "actor")
    private List<ClassesActor> actorClasses;
	
	//getter和setter方法
	}

ClassesActor类

@Entity
public class ClassesActor {

    private Long id;

    @ManyToOne
    @JoinColumn(name = "actor_id")
    private Actor actor;

    @ManyToOne
    @JoinColumn(name = "class_id")
    private Classes classes;

	//getter和setter方法
	}

  当我在实际的业务中去查询classes相关信息的时候会发现,Classes对象中有ClassesActor对象,点开ClassesActor对象中海油Classes对象而且这些对象都是同一个对象。所以就出现了循环嵌套的问题。
在这里插入图片描述

解决问题

分析问题

   1. 我们要清楚对于对象的属性进行操作属于数据序列化的过程。数据序列化是将对象转换为字节流或其他形式的数据,以便在网络传输、存储或跨平台传递时使用。在数据序列化过程中,对象的属性值会被转换为字节流或其他格式的数据,并随后可以被反序列化为对象。
  2.通过对问题的分析我们可以通过序列化操作来解决序列化多个相同对象的问题,这样我们就把问题缩小到序列化中。
  3.明确如何在序列化的过程中去控制相同的对象只序列化一次,我们的问题也就迎刃而解了。

具体解决

  通过@JsonManagedReference和@JsonBackReference注解帮助我们解决相同的对象只序列化一次的问题。

注解说明

  • 「@JsonManagedReference 」注解用于标注在实体类的属性上,表示该属性是一个“被管理的引用”。它的作用是告诉Jackson在序列化过程中,该属性是“正向”引用,需要被序列化输出。同时,它还需要配合@JsonBackReference注解一起使用,指定“反向”引用的属性。
  • 「@JsonBackReference 」 注解用于标注在实体类的属性上,表示该属性是一个“反向引用”。它的作用是告诉Jackson在序列化过程中,该属性是“反向”引用,不需要被序列化输出。相反,它会通过@JsonManagedReference注解指定的属性来进行序列化输出。

代码实现

Classes类

@Service
@Entity
@Table(name="Classes")
public class Classes implements ApplicationRunner {

    private Long id;

    private Long classId;

    private String className;

    private Integer isDelete;


    @OneToMany(mappedBy = "classes")
	@JsonManagedReference(value = "class-actorClasses")
    private List<ClassesActor> actorClasses;

	//getter和setter方法
	
	}

Actor类

@Service
@Entity
@Table(name = "Actor")
public class Actor  {
    private Long id;

    private String code;

    private String name;

    private String phone;

	@OneToMany(mappedBy = "actor")
	@JsonManagedReference(value = "actor-classes")
    private List<ClassesActor> actorClasses;
	
	//getter和setter方法
	}

ClassesActor类

@Entity
public class ClassesActor {

    private Long id;

    @ManyToOne
    @JoinColumn(name = "actor_id")
    @JsonBackReference(value = "actor-classes")
    private Actor actor;

    @ManyToOne
    @JoinColumn(name = "class_id")
    @JsonBackReference(value = "class-actorClasses")
    private Classes classes;

	//getter和setter方法
	}

  通过使用@JsonManagedReference和@JsonBackReference注解,Jackson库能够正确处理实体类之间的循环引用关系,避免了无限递归的序列化问题。当一个类中出现了多个@JsonBackReference注解要有value值进行区分和加以对应。

效果展示

在这里插入图片描述

总结提升

  这两个注解只适用于序列化过程,对于反序列化过程是不起作用的。如果需要在反序列化时处理循环引用问题,可以考虑使用@JsonIdentityInfo注解或自定义序列化和反序列化逻辑来处理。



在这里插入图片描述


🎯 此文章对你有用的话记得留言+点赞+收藏哦🎯
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

武梓龙_Wzill

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值