【java框架】JPA(3) -- JPA映射關係

来源:https://www.cnblogs.com/yif0118/archive/2020/05/19/12914104.html
-Advertisement-
Play Games

1. 單向一對多配置 單向一對多使用@OneToMany標簽進行配置,在一方有一個集合屬性與多方進行關聯,集合可以是List或者Set,區別是List是有序、Set是無序不重覆。 對應在一方配置@OneToMany: /** * 單向一對多:使用JPA配置 */ @Entity @Table(nam ...


1.   單向一對多配置

單向一對多使用@OneToMany標簽進行配置,在一方有一個集合屬性與多方進行關聯,集合可以是List或者Set,區別是List是有序、Set是無序不重覆。

對應在一方配置@OneToMany:

/**
 *  單向一對多:使用JPA配置
 */
@Entity
@Table(name = "t_productdir")
public class ProductDir {
    @Id
    @GeneratedValue
    private Integer id;

    @Column(name = "dName")
    private String dirName; //產品分類名稱

    /**
     *  單向一對多:在一方使用集合Set或者List進行多方關係的維護
     *  對於集合,必須要先new出來
     *
     *  對於資料庫:不管是多對一,還是一對多,不管是單向還是雙向,
     *  資料庫的設置都是不變的,哪邊是多方,外鍵就在哪邊
     *  單向一對多,預設就是使用的懶載入(以後都把懶載入配置上)
     */
    @OneToMany(fetch = FetchType.LAZY)
    @JoinColumn(name = "dir_id")
    @OrderBy("id desc") //需要排序的時候使用@OrderBy來拿值,多個排序屬性之間使用","分割,並且一定要用List集合
    private Set<Product> productSet = new HashSet<>(); //集合預設懶載入

    public ProductDir() {
    }

性能:單向一對多無論配置懶載入還是迫切載入都發送相同數量的SQL語句,性能極差。

2.   雙向一對多、多對一配置

2.1.基礎配置

雙向一對多、多對一需要同時配置兩邊的屬性,一方與多方都要有關聯屬性存在(同時配置@ManyToOne、@OneToMany),同時在一方放棄關係維護配置mappedBy。具體配置如下:

多方Product:

/**
 *  雙向一對多、多對一:使用JPA配置
 */
@Entity
@Table(name = "t_product")
public class Product {
    @Id
    @GeneratedValue
    private Integer id;

    @Column(name = "t_name")
    private String name; //產品名稱

    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "dir_id")
    private ProductDir productDir;

    public ProductDir getProductDir() {
        return productDir;
    }

    public void setProductDir(ProductDir productDir) {
        this.productDir = productDir;
    }

    public Product() {
    }

}

一方ProductDir:

/**
 *  雙向一對多、多對一:使用JPA配置
 */
@Entity
@Table(name = "t_productdir")
public class ProductDir {
    @Id
    @GeneratedValue
    private Integer id;

    @Column(name = "dName")
    private String dirName; //產品分類名稱

    //mappedBy:一方放棄關係維護,把關係維護交給多方,註意mappedBy中的值必須和Product中ProductDir的屬性一樣
    //註意:使用mappedBy就不要使用JoinColumn了,這邊已經不需要維護關係了
    @OneToMany(fetch = FetchType.LAZY, mappedBy = "productDir")
    //@JoinColumn(name = "dir_id")
   // @OrderBy("id desc") //需要排序的時候使用@OrderBy來拿值,多個排序屬性之間使用","分割,並且一定要用List集合
    private Set<Product> productSet = new HashSet<>(); //集合預設懶載入

    public ProductDir() {
    }

}

2.2.級聯配置

在一方的多方屬性@OneToMany上使用cascade表示級聯,主要有以下幾種配置方式:

①    Cascade=CascadeType.PERSIST:級聯保存;

②    Cascade=CascadeType.REMOVE:級聯刪除(很危險);

③    Cascade=CascadeType.ALL:級聯增刪改

3.   單向多對多

多對多關係我們以保存2個用戶(user)、3個角色(role)來進行多對多的測試;

用戶與角色是多對多的關係,一個用戶對應多個角色,一個角色可以由多個用戶充當。

多堆多關係涉及到一張中間表user_role,

單向關係:通過用戶可以找到多個角色,而角色不能找到對應的用戶。

3.1.單向多對多配置

主表:User用戶類

/**
 * 基於單向多對多,在用戶方可以找到對應的角色(角色不能找到對應用戶),只需要在用戶方配置@ManyToMany註解映射
 */
@Entity
@Table(name="t_user")
public class User {
    @GeneratedValue
    @Id
    private Long id;

    private String name;

    // @ManyToMany註釋表示User是多對多關係的一端。
    // @JoinTable描述了多對多關係的中間表關係。name屬性指定中間表名稱,
    // joinColumns定義中間表與當前類User的外鍵關係。inverseJoinColumns定義中間表與關聯類Role的外鍵關係。
    @ManyToMany
    @JoinTable(name = "t_user_role", joinColumns = { @JoinColumn(name = "user_id") }, inverseJoinColumns = {
            @JoinColumn(name = "role_id") })
    private Set<Role> roles = new HashSet<Role>();

    public User() {
    }

    public User(String name) {
        this.name = name;
    }
}

從表:Role角色類

@Entity
@Table(name="t_role")
public class Role {
    @GeneratedValue
    @Id
    private Long id;

    private String name;

    public Role() {
    }

    public Role(String name) {
        this.name = name;
    }
}

對應創建測試類:

/**
     * 保存2個用戶,保存3個角色(5條)
     */
    @Test
public void persistUserTest() {
    User user1 = new User("user1");
    User user2 = new User("user2");

    Role role1 = new Role("role1");
    Role role2 = new Role("role2");
    Role role3 = new Role("role3");

    // 保存中間表:建立用戶到角色關係user1(role1,role2),user2(role1,role2,role3)(5條)
    user1.getRoles().add(role1);
    user1.getRoles().add(role2);
    user2.getRoles().add(role1);
    user2.getRoles().add(role2);
    user2.getRoles().add(role3);

    EntityManager entityManager = JPAUtil.getEntityManager();
    entityManager.getTransaction().begin();

    //保存用戶
    entityManager.persist(user1);
    entityManager.persist(user2);
    // 保存角色
    entityManager.persist(role1);
    entityManager.persist(role2);
    entityManager.persist(role3);

    entityManager.getTransaction().commit();
    JPAUtil.close(entityManager);
}

4.   雙向多對多

4.1.雙向多堆多配置

雙向多對多即是在兩方同時配置@ManyToMany註解,具體配置如下:

一方Role角色配置:

@Entity
@Table(name="t_role")
public class Role {
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Id
    private Long id;

    private String name;

    // @ManyToMany註釋表示Role是多對多關係的一端。
    @ManyToMany
    //joinColumns表示中間表與當前Role表的連接外鍵,inverseJoinColumns表示多方User與中間表的外鍵連接關係
    @JoinTable(name = "t_user_role", joinColumns = { @JoinColumn(name = "role_id") }, inverseJoinColumns = {
            @JoinColumn(name = "user_id") })
    private Set<User> userSet = new HashSet<>();

    public Role() {
    }

    public Set<User> getUserSet() {
        return userSet;
    }

    public void setUserSet(Set<User> userSet) {
        this.userSet = userSet;
    }

    public Role(String name) {
        this.name = name;
    }

}

多方User用戶配置:

/**
 * 基於單向多對多,在用戶方可以找到對應的角色(角色不能找到對應用戶),只需要在用戶方配置@ManyToMany註解映射
 */
@Entity
@Table(name="t_user")
public class User {
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Id
    private Long id;

    private String name;

    // @ManyToMany註釋表示User是多對多關係的一端。
    // @JoinTable描述了多對多關係的中間表關係。name屬性指定中間表名稱,
    // joinColumns定義中間表與當前類User的外鍵關係。inverseJoinColumns定義中間表與關聯類Role的外鍵關係。
    @ManyToMany
    @JoinTable(name = "t_user_role", joinColumns = { @JoinColumn(name = "user_id") }, inverseJoinColumns = {
            @JoinColumn(name = "role_id") })
    private Set<Role> roles = new HashSet<Role>();

    public User() {
    }

    public User(String name) {
        this.name = name;
    }

}

5.   一對一配置

一對一場景在實際項目中用得比較少,這裡主要以QQ與QQ空間為例,有QQ才會有QQ空間。

5.1.唯一外鍵一對一

QQ類(主一)

@Entity
public class QQ {
  @Id
  @GeneratedValue
  private Long id;
  private String number;
  // 一對一,一個qq號碼對應一個qq空間
  @OneToOne(mappedBy="qq")
  private QQZone zone;
}

QQ空間類(從一)

@Entity
public class QQZone {
  @Id
  @GeneratedValue
  private Long id;
  private String name;
  // 一對一,一個qq空間屬於一個qq號碼
  // 預設值optional = true表示qq_id可以為空;反之。。。
  @OneToOne(optional = false)
  // unique=true確保了一對一關係
  @JoinColumn(name = "qq_id", unique = true)
  private QQ qq;
}

5.1.1保存數據測試

public void persist() throws Exception {
    QQ qq = new QQ();
    qq.setNumber("123456");

    QQZone zone = new QQZone();
    zone.setName("楓夜");

    // 建立關係
    qq.setZone(zone);
    zone.setQq(qq);

    EntityManager entityManager = JPAUtils.getEntityManager();
    entityManager.getTransaction().begin();

    // 先保存主一
    entityManager.persist(qq);
    entityManager.persist(zone);

    entityManager.getTransaction().commit();
    entityManager.close();
}

5.2.共用主鍵一對一

QQ空間類(從一)

@Entity
public class QQZone {
  @Id
  @GeneratedValue(generator = "fkGenerator")
  @GenericGenerator(name = "fkGenerator", strategy = "foreign", parameters = @Parameter(name = "property", value = "qq"))
  private Long id;
  private String name;
  // 一對一,一個qq空間輸入一個qq號碼
  @OneToOne(optional = false)
  // 如果不加這個註解,添加QQZone信息時,就會自動在QQZone表中增加了一個外鍵qq_id
  @PrimaryKeyJoinColumn
  private QQ qq;
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本項目演示地址:https://husilang.github.io/zm-ui 項目參考文章:從零開始搭建Vue組件庫 VV-UI 項目的初衷是學習怎麼封裝一個基於Vue的UI組件庫,順便記錄每個步驟,以及在此過程中遇到的難點及體會。 下麵是我個人的一個項目搭建流程,希望能幫助大家。 ①腳手架初始 ...
  • # 3.border和background - border-radius border-radius:上左 上右 下右 下左 border-radius:上左下右 上右下左 border-radius:上左 上右下左 下右 border-top-left-radius:10px 10px; bor ...
  • 常用 "CSS選擇器" 查看地址 (https://www.cnblogs.com/hleisurely/p/12836777.html) 1.選擇器的優先順序 !important的優先順序別最高 權重值為1111 用來提升某個直接選中標簽的選擇器中某個屬性的優先順序的,可以將被指定的屬性的優先順序提升為 ...
  • 前言 Deno 已經正式發佈了! 我說這句話時候,是不是很多前端 和 NodeJS 工(碼)程(農)師已經按不住自己的40米大刀了。心中的不僅感慨前端是真的會造輪子,有了 node 還不夠嗎,還沒學會 node 又搞了個 deno,node 和 deno 啥區別?! 的確,deno 和 node 形 ...
  • 今天給大家整理出來了web前端工程師初級階段需要掌握的內容,很全面,希望大家好好閱讀,看看自己掌握的知識點和文章裡面寫的還相差多少。 什麼是初級web前端工程師? 初級前端工程師:首先要知道的就是如何處理各種瀏覽器的相容處理(比如說在IE瀏覽器中的createElement有什麼不同等等內容),現在 ...
  • 一、ajax 輪詢實現原理:ajax 輪詢指客戶端每間隔一段時間向服務端發起請求,保持數據的同步。 優點:可實現基礎(指間隔時間較短)的數據更新。 缺點:這種方法也只是儘量的模擬即時傳輸,但並非真正意義上的即時通訊,很有可能出現客戶端請求時,服務端數據並未更新。或者服務端數據已更新,但客戶端未發起請 ...
  • 一、發展由來 POP OOP AOP DDD POP:(面向過程編程) 描述:考慮問題是1234步驟 特點:簡單,無法應對複雜需求 舉例子:把食物放冰箱 第一步:打開冰箱 第二步:塞進去食物 第三步:關冰箱 OOP:面向對象編程 特點:把事物作為對象考慮 特性:封裝、繼承、多態 描述:對象交互,完成 ...
  • 使用CLion替換VSCode,開發 chromium kernel(for Linux) VSCode 不適合開發像chromium 這樣的巨型c++工程,Microsoft的cpptools和mono(.net移植,被VScode用作來寫code intellisense)存在的嚴重的memor ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...