Spring Data JPA關係映射@OneToOne

@OneToOne, 表示一對一的映射關係,比如一個賬號對應一個用戶,一個實體用來描述賬號的信息(賬號,密碼,賬號是否可用,賬號對應的角色等),另外一個實體用來描述用戶的信息(暱稱,年齡,性別,國籍等)。

該註解有六個屬性:

public @interface OneToOne {
    java.lang.Class targetEntity() default void.class;

    javax.persistence.CascadeType[] cascade() default {};

    javax.persistence.FetchType fetch() default javax.persistence.FetchType.EAGER;

    boolean optional() default true;

    java.lang.String mappedBy() default "";

    boolean orphanRemoval() default false;
}
  1. targetEntity 關聯目標實體類,指定類型後該屬性可省略;
  2. cascade表示關聯關係中的級聯操作權限,有五種權限:

    • CascadeType.PERSIST:級聯新增(又稱級聯保存);
    • CascadeType.MERGE:級聯合並,更新該實體時,與其有映射關係的實體也跟隨更新;
    • CascadeType.REMOVE:級聯刪除,刪除該實體時,與其有映射關係的實體也跟隨刪除;
    • CascadeType.REFRESH:級聯刷新,該實體被操作前都會刷新,保證數據合法性;
    • CascadeType.ALL:包含以上四種級聯操作;
  3. fetch數據加載策略,默認值爲FetchType.EAGER

    • FetchType.LAZY 表示數據獲取方式爲懶加載;
    • FetchType.EAGER 表示數據獲取方式爲急加載;
  4. optional 表示關聯關係是否必須,當該值爲true時,one的一方可以爲null
  5. mappedBy 指定映射關係由哪一方維護,一般使用在雙向映射場景;
  6. orphanRemoval 孤值刪除,將會刪除孤立數據,外鍵爲null的數據將被刪除;

我們在使用的時候,通常爲了保證表的簡潔性,將主鍵共享,意思是用戶的id和賬號的id是一樣的,不在表中單獨存在一個字段用來描述關聯關係;比如下面的例子:
首先創建一個賬號實體

import org.hibernate.annotations.GenericGenerator;
import org.hibernate.annotations.Parameter;
import javax.persistence.*;

@Table(name = "base_account")
@Entity
@org.hibernate.annotations.Table(appliesTo = "base_account", comment = "賬號信息表")
public class AccountDO {

    @Id
    @GenericGenerator(name="idGenerator", strategy = "uuid")
    @GeneratedValue(generator = "idGenerator")
    @Column(name = "ACCOUNT_ID", length = 32)
    private String accountId;

    @Column(name = "USERNAME", columnDefinition = "VARCHAR(32) NOT NULL COMMENT '賬號'")
    private String username;

    @Column(name = "PASSWORD", columnDefinition = "VARCHAR(128) NOT NULL COMMENT '密碼'")
    private String password;

    @OneToOne(cascade = {CascadeType.PERSIST, CascadeType.REMOVE, CascadeType.REFRESH})
    @PrimaryKeyJoinColumn
    private UserDO userDO;
    
    // 省略構造函數,get/set方法,toString方法等

創建一個用戶信息實體

import org.hibernate.annotations.GenericGenerator;
import org.hibernate.annotations.Parameter;
import javax.persistence.*;

@Table(name = "base_user")
@Entity
@org.hibernate.annotations.Table(appliesTo = "base_user", comment = "用戶信息表")
public class UserDO {

    @Id
    @GenericGenerator(name = "idGenerator", strategy = "foreign", parameters = @Parameter(name = "property", value = "accountDO"))
    @GeneratedValue(generator = "idGenerator")
    @Column(name = "USER_ID", length = 32)
    private String userId;

    @Column(name = "NICKNAME", columnDefinition = "VARCHAR(32) NOT NULL COMMENT '暱稱'")
    private String nickname;

    @Column(name = "AGE", columnDefinition = "TINYINT DEFAULT NULL COMMENT '年齡'")
    private Integer age;

    @Column(name = "SEX", columnDefinition = "CHAR(2) DEFAULT NULL COMMENT '性別'")
    private String sex;

    @OneToOne(mappedBy = "userDO")
    private AccountDO accountDO;
    
        // 省略構造函數,get/set方法,toString方法等

用戶實體的主鍵和賬號實體的主鍵都使用一個生成策略,生成的id也一樣,且在賬號實體中使用@PrimaryKeyJoinColumn來聲明在表中不建立對應的映射字段。

這裏貼出源碼,一個關係映射的小例子

原創不易,感謝支持。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章