news 2026/4/18 13:56:47

JPA多對多關係時 JSON 序列化解决方案

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
JPA多對多關係時 JSON 序列化解决方案

前言

在 JPA 中處理 多對多 (Many-to-Many) 關係,不使用 @ManyToMany 註解方式,而是將這個關係拆解為兩個一對多的單向關係,並為中間表創建一個獨立的Entity.

代碼如下:

@Entity @Data @NoArgsConstructor @AllArgsConstructor @Builder @Table(name = "users") public class User { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Long id; @Column(name = "username", nullable = false) private String username; @Column(name = "password", nullable = false) private String password; @Column(name = "first_name", nullable = false) private String firstName; @Column(name = "last_name") private String lastName; @Column(name = "email", nullable = false, unique = true) private String email; @OneToMany(mappedBy = "user", cascade = CascadeType.ALL, orphanRemoval = true) private Set<UserRole> roles = new HashSet<>(); public User(String username, String password, String firstName, String lastName, String email) { this.username = username; this.password = password; this.firstName = firstName; this.lastName = lastName; this.email = email; } }
@Entity @Data @NoArgsConstructor @AllArgsConstructor @Builder @Table(name = "roles") public class Role { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Long id; @Column(length = 20) private String name; public Role(String name) { this.name = name; } @OneToMany(mappedBy = "role", cascade = CascadeType.ALL, orphanRemoval = true) private Set<UserRole> userRoles = new HashSet<>(); }
@Data @Embeddable public class UserRoleId implements Serializable { // 與 UserRole.java 中 @MapsId 的名稱一致 @Column(name = "user_id") private Long userId; @Column(name = "role_id") private Long roleId; public UserRoleId() { } public UserRoleId(Long userId, Long roleId) { this.userId = userId; this.roleId = roleId; } @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; UserRoleId that = (UserRoleId) o; return Objects.equals(userId, that.userId) && Objects.equals(roleId, that.roleId); } @Override public int hashCode() { return Objects.hash(userId, roleId); } }
@Entity @Data @NoArgsConstructor @AllArgsConstructor @Builder @Table(name = "users_roles") public class UserRole implements Serializable { // ID @EmbeddedId private UserRoleId id; // 關係到 User, userId 對映到 UserRoleId 中的 userId @ManyToOne(fetch = FetchType.LAZY) @MapsId("userId") @JoinColumn(name = "user_id") private User user; // 關係到 Role @ManyToOne(fetch = FetchType.LAZY) @MapsId("roleId") @JoinColumn(name = "role_id") private Role role; @Column(name = "assigned_at") private LocalDateTime assignedAt; }

當我們 序列化 User 實例時,Jackson 會拋出JsonMappingException異常

顯示Exception如下:

原因:

這個錯誤發生在 Jackson 嘗試將您的 JPA 實體 User 序列化為 JSON 字串時,Jackson 序列化器仍然發現了一個循環

無限遞歸序列化錯誤原因:

循環序列化的多對多結構 User <-> UserRole <-> Role

  1. Jackson 序列化 User。
  2. 在序列化 User 的屬性時,遇到 roles 集合 (Set<UserRole>)。
  3. 序列化 UserRole 時,遇到 User 實體 (@ManyToOne private User user;)。
  4. Jackson 再次嘗試序列化這個 User 物件,回到步驟 1,形成無限循環。

註: Jackson 預設的最大遞歸深度是 1000 層,當達到這個限制時,它會拋出這個錯誤以避免堆棧溢出(StackOverflowError)。

任務

針對 User 序列化為 JSON 字串時,Jackson JSON 的無限遞迴問題,提出處理雙向關係的方法

處理動作

步驟一. 首先建立一個測試案例測試:

@Transactional @SpringBootTest public class UserRoleRelationshipTest { @Test void testReadUserRoleRelationship() { try { List<User> users = userRepository.findAll(); // 獲取所有用戶 System.out.println("****** 獲取所有用戶: ******"); ObjectMapper objectMapper = new ObjectMapper(); objectMapper.registerModule(new JavaTimeModule()); String jsonArray = objectMapper.writerWithDefaultPrettyPrinter().writeValueAsString(users); System.out.println(jsonArray); } catch (JsonProcessingException e) { e.printStackTrace(); } }

步驟二. 預備測試資料,已存 DB

測試用

Table users

步驟三. 實作方案

方案一:使用@JsonIgnore

不想序列化某個屬性,使用@JsonIgnore註解來忽略關係中的某個屬性

選項1

public class User { . . . @OneToMany(mappedBy = "user", cascade = CascadeType.ALL, orphanRemoval = true) @JsonIgnore private Set<UserRole> roles = new HashSet<>(); . . . }

測試結果

% mvn test

選項2

public class UserRole implements Serializable { . . . @ManyToOne(fetch = FetchType.LAZY) @MapsId("userId") // 映射到 UserRoleId 中的 userId @JoinColumn(name = "user_id") @JsonIgnore private User user; @ManyToOne(fetch = FetchType.LAZY) @MapsId("roleId") // 映射到 UserRoleId 中的 roleId @JoinColumn(name = "role_id") @JsonIgnore private Role role; . . . }

測試結果

% mvn test

方案二:使用@JsonManagedReferences@JsonBackReferences

@OneToMany(mappedBy = "user", cascade = CascadeType.ALL, orphanRemoval = true) @JsonManagedReference private Set<UserRole> roles = new HashSet<>(); @OneToMany(mappedBy = "role", cascade = CascadeType.ALL, orphanRemoval = true) @JsonManagedReference private Set<UserRole> userRoles = new HashSet<>(); @ManyToOne(fetch = FetchType.LAZY) @MapsId("userId") // 映射到 UserRoleId 中的 userId 屬性 @JoinColumn(name = "user_id") @JsonBackReference private User user; @ManyToOne(fetch = FetchType.LAZY) @MapsId("roleId") // 映射到 UserRoleId 中的 roleId 屬性 @JoinColumn(name = "role_id") @JsonBackReference private Role role;

執行測試結果:

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/18 8:34:17

Stop-motion-OBJ:Blender网格序列动画导入终极指南

Stop-motion-OBJ&#xff1a;Blender网格序列动画导入终极指南 【免费下载链接】Stop-motion-OBJ A Blender add-on for importing a sequence of OBJ meshes as frames 项目地址: https://gitcode.com/gh_mirrors/st/Stop-motion-OBJ Stop-motion-OBJ是一款功能强大的B…

作者头像 李华
网站建设 2026/4/18 5:41:30

20、Linux 系统软件更新与安装全攻略

Linux 系统软件更新与安装全攻略 在 Linux 系统中,软件的更新和安装是日常使用中常见的操作。下面将详细介绍不同 Linux 发行版下软件更新和安装的方法。 软件更新 在某些情况下,完成软件更新需要以下步骤: 1. 输入管理员/根用户密码并按回车键,更新开始。更新完成后,…

作者头像 李华
网站建设 2026/4/18 7:37:31

25、Linux办公软件指南:OpenOffice.org全解析

Linux办公软件指南:OpenOffice.org全解析 1. 软件基础介绍 在如今的计算机世界里,办公软件是每个人都不可或缺的工具。对于Windows用户来说,Microsoft Office是常见的选择,但在Linux系统中,OpenOffice.org则是主流的办公套件。它包含了Base(数据库)、Calc(电子表格)…

作者头像 李华
网站建设 2026/4/18 6:27:33

27、Linux 系统的打印配置与多媒体使用指南

Linux 系统的打印配置与多媒体使用指南 1. OpenOffice.org 打印配置 在使用 OpenOffice.org 进行打印之前,可能需要对其进行打印设置。若之前已完成打印设置,可尝试从 OpenOffice.org 的任意应用程序打印一页,查看是否能正常工作。对于无法从 OpenOffice.org 打印,但能从…

作者头像 李华
网站建设 2026/4/18 6:30:00

[CISCN2019 华东南赛区]Web111

打开题目&#xff0c;看到末尾 Build With Smarty ! 推测 Smarty 模板注入 看了wphttps://blog.csdn.net/kw741951/article/details/141001010 smart是php的模板引擎&#xff0c;模板引擎的作用就是分离前端页面和数据的&#xff0c;题目中显示API的URL由于环境的原因无法使…

作者头像 李华