【Java关系映射入门】实战一
系列文章目录
Java实战知识
文章目录
- 系列文章目录
- 👉前言
- 👉一、@ManyToOne (多对一)
- 👉1-1、示例
- 👉1-2、注意事项
- 👉二、@OneToMany (一对多)
- 👉2-1、示例
- 👉2-2、注意事项
- 👉壁纸分享
- 👉总结
👉前言
理解 Java 中的关系映射(JPA/Hibernate)是构建数据库驱动应用的核心。下面详细解析 @OneToOne, @OneToMany, @ManyToOne, @ManyToMany 的原理、使用及关键注意事项:
核心原理: 这些注解将对象间的关联关系(基于面向对象)映射到数据库表间的关联关系(基于关系模型)。ORM 框架(如 Hibernate)负责在运行时根据这些注解生成 SQL 语句(JOIN, 子查询等)来加载或保存相关联的数据。
博客将会介绍如何实现Java关系映射。希望这篇博客对Unity的开发者有所帮助。
大家好,我是心疼你的一切,不定时更新Unity开发技巧,觉得有用记得一键三连哦。
欢迎点赞评论哦.下面就让我们进入正文吧 !
提示:以下是本篇文章正文内容,下面案例可供参考
👉一、@ManyToOne (多对一)
原理:
-
概念: 表示“多”方实体引用“一”方实体。这是最常见的关联,也是构建其他关联的基础。
-
数据库体现: “多”方对应的表拥有一个外键列,指向“一”方表的主键。
-
使用: 在“多”方实体类中,使用 @ManyToOne 标注关联字段。
-
关键属性:
fetch:加载策略 (FetchType.LAZY - 懒加载 [推荐默认], FetchType.EAGER - 急加载)。
cascade:级联操作类型 (CascadeType.PERSIST, CascadeType.MERGE, CascadeType.REMOVE 等),决定在操作当前实体时是否级联操作关联实体。
optional:关联是否可为空 (true - 可为空 [默认], false - 不可为空)。
👉1-1、示例
多个 Order (订单) 属于一个 Customer (客户)。
代码如下:
@Entity
public class Order {@Idprivate Long id;@ManyToOne(fetch = FetchType.LAZY) // 推荐使用懒加载@JoinColumn(name = "customer_id") // 指定外键列名,可选private Customer customer; // 引用“一”方// ... other fields, getters, setters
}@Entity
public class Customer {@Idprivate Long id;// ... other fields, getters, setters// 通常不需要在这里定义反向的 @OneToMany,除非需要双向关联
}
👉1-2、注意事项
-
外键管理: @ManyToOne 端是关系的拥有方。数据库外键列存在于 Order 表中。
-
懒加载: 强烈建议使用 FetchType.LAZY。加载一个 Order 时,默认不会立即加载关联的 Customer,避免不必要的数据加载和性能开销(N+1 查询问题通常由错误使用 EAGER 引起)。当首次访问 order.getCustomer() 时,Hibernate 才会去查询数据库加载 Customer。
-
级联: 谨慎使用级联删除 (CascadeType.REMOVE)。删除一个 Order 不应该级联删除它的 Customer(其他订单可能还属于该客户)。级联保存 (CascadeType.PERSIST) 通常更安全。
-
索引: 确保外键列 (customer_id) 上有索引,这对关联查询性能至关重要。
👉二、@OneToMany (一对多)
原理:
-
概念: 表示“一”方实体拥有一个“多”方实体的集合。通常与 @ManyToOne 构成双向关联。
-
数据库体现: 关系由“多”方表的外键维护(与 @ManyToOne 相同)。@OneToMany 本身不直接在数据库创建新结构,它定义了“一”方如何访问基于外键关联的“多”方集合。
-
使用: 在“一”方实体类中,使用 @OneToMany 标注集合字段。必须使用 mappedBy 属性(在双向关联中)指定反向关联字段(即“多”方中指向“一”方的字段)。
-
关键属性:
mappedBy:(双向关联必备) 指定关联关系在“多”方实体中的哪个字段进行映射(即“多”方中 @ManyToOne 或 @OneToOne 的字段名)。这表示关系的维护权交给了“多”方。
fetch:加载策略 (LAZY [集合默认], EAGER)。
cascade:级联操作类型。
orphanRemoval:是否自动删除不再被集合引用的“孤儿”实体 (true/false)。
👉2-1、示例
一个 Customer 拥有多个 Order (双向关联)。
代码如下:
@Entity
public class Customer {@Idprivate Long id;@OneToMany(mappedBy = "customer", // 指向 Order 中的 customer 字段fetch = FetchType.LAZY, // 集合默认 LAZY 是好的cascade = CascadeType.ALL, // 级联保存/更新/删除 Order?谨慎考虑删除!orphanRemoval = true) // 从集合中移除的 Order 会被自动删除private List<Order> orders = new ArrayList<>();// ... other fields, getters, setters// 辅助方法(维护双向关联一致性)public void addOrder(Order order) {orders.add(order);order.setCustomer(this);}public void removeOrder(Order order) {orders.remove(order);order.setCustomer(null);}
}@Entity
public class Order {@Idprivate Long id;@ManyToOne(fetch = FetchType.LAZY)@JoinColumn(name = "customer_id")private Customer customer; // mappedBy 指向这里// ... other fields, getters, setters
}
👉2-2、注意事项
-
mappedBy 是核心: 在双向 @OneToMany 中,必须使用 mappedBy 并将关系的维护权交给 @ManyToOne 端(即外键拥有者)。没有 mappedBy 意味着双方都试图维护关系,会导致重复更新或额外中间表(Hibernate 默认行为)。
-
集合初始化: 将集合初始化为空集合 (new ArrayList<>(), new HashSet<>()),避免 null。
-
双向关联一致性: 强烈推荐在“一”方 (Customer) 中添加辅助方法 (addOrder, removeOrder) 来同时操作双方的引用,确保内存中对象状态的一致性。直接操作 orders.add(order) 而不设置 order.setCustomer(this) 会导致状态不一致。
-
orphanRemoval: 启用后,当你从 Customer 的 orders 集合中移除一个 Order 并保存 Customer 时,被移除的 Order 会被自动删除。这适用于组合关系(Order 是 Customer 的一部分,没有独立存在意义)。慎用,确保业务逻辑符合预期。
-
级联: 级联 CascadeType.ALL 通常对 @OneToMany 更常见(保存 Customer 时自动保存其所有 Order)。但极其谨慎级联删除 (REMOVE 或 ALL),删除一个 Customer 会删除其所有 Order!确保这是业务需求。通常业务上可能只做逻辑删除或转移订单。
-
懒加载: 集合默认是 LAZY 加载,这是正确的。 加载 Customer 时不会立即加载所有 Order。访问 customer.getOrders() 时才会触发加载。N+1 查询问题最常见于遍历这种懒加载集合(在循环外加载或使用 JOIN FETCH 解决)。
👉壁纸分享
👉总结
本次总结的就是 Java关系映射的实现, 有需要会继续增加功能
如能帮助到你,就帮忙点个赞吧,三连更好哦,谢谢
你的点赞就是对博主的支持,有问题记得留言评论哦!
不定时更新Unity开发技巧,觉得有用记得一键三连哦。么么哒!