首先,实体与实体之间的关联关系一共分为四种,分别为OneToOne、OneToMany、ManyToOne和ManyToMany;而实体之间的关联关系又分为 双向和单向
。实体之间的关联关系是在JPA使用中最容易发生问题的地方。
1、OneToOne关联关系
@OneToOne一般表示对象之间一对一的关联关系,它可以放在field上面,也可以放在get/set方法上面。其中JPA协议有规定, 如果配置双向关联,维护关联关系的是拥有外键的一方,而另一方必须配置mappedBy
;如果是单项关联,直接配置在拥有外键的一方即可。
举例说明:
user表是用户的主信息,user_info是用户的拓展信息, 两者之间是一对一的关系
。user_info表里面有一个user_id作为关联关系的外键, 如果是单项关联
,我们的写法如下:
@Data @Entity @NoArgsConstructor @AllArgsConstructor @Builder public class User { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Integer id; private String name; private String email; private String sex; private String address; } 复制代码
我们只需要在拥有外键的一方 配置@OneToOne注解
就可以了
@Entity @Data @Builder @AllArgsConstructor @NoArgsConstructor @ToString(exclude = "user") public class UserInfo { @Id @GeneratedValue(strategy = GenerationType.AUTO) private Integer id; private Integer ages; private String telephone; @OneToOne private User user; } 复制代码
这就是 单向关联关系
,那么如何设置 双向关联关系
呢? 我们保持UserInfo不变,在User实体对象里面添加一段代码即可
@OneToOne(mappedBy = "user") private UserInfo userInfo; @Data @Entity @NoArgsConstructor @AllArgsConstructor @Builder public class User { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) @Column(name = "id") private Integer id; private String name; private String email; private String sex; private String address; @OneToOne(mappedBy = "user") private UserInfo userInfo; } 复制代码
1.1 解读OneToOne源码
public @interface OneToOne { Class targetEntity() default void.class; CascadeType[] cascade() default {}; FetchType fetch() default EAGER; boolean optional() default true; String mappedBy() default ""; boolean orphanRemoval() default false; } 复制代码
targetEntity:作为关联目标的实体类。
cascade:级联操作策略,就是我们常说的级联操作。
fetch:数据获取方式EAGER(立即加载)/LAZY(延迟加载) optional :表示关联的实体是否能够存在null值 mappedBy :关联关系被谁维护的一方对象里面的属性名字, 双向关联的时候必填
。
1.2 mappedBy 注意事项
-
只有
关联关系的维护方
才能操作两个实体之间外键的关系。被维护方即使设置维护方属性进行存储也不会更新外键关联
-
mappedBy不能与@JoinColumn或者@JoinTable同时使用,因为没有任何意义,关联关系不在这里面维护。
-
mappedBy的值是指
另一方的实体里面属性的字段
,而不是数据库字段,也不是实体的对象的名字。也就是维护关联关系的一方属性字段名称
,或者加了@JoinColumn 或 @JoinTable注解的属性字段名称。如上面的User例子user里面