欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

DoytoQuery中關(guān)于N+1查詢問題解決方案詳解

 更新時(shí)間:2022年12月27日 17:15:00   作者:f0rb  
這篇文章主要為大家介紹了DoytoQuery中關(guān)于N+1查詢問題解決方案詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

1. 背景

Java Persistence with Hibernate 在12.2.1小節(jié)使用如下例子描述 n+1查詢問題:

List<Item> items = em.createQuery("select i from Item i").getResultList();
// select * from ITEM
for (Item item : items) {
    assertTrue(item.getBids().size() > 0);
    // select * from BID where ITEM_ID = ?
}

在這個(gè)例子中,每個(gè)bids集合的加載都需要執(zhí)行一條額外的查詢語句,當(dāng)item有N條記錄,一共就會(huì)執(zhí)行N+1條查詢語句:

SELECT * FROM item;
SELECT * FROM bid WHERE item_id = ?;
SELECT * FROM bid WHERE item_id = ?;
SELECT * FROM bid WHERE item_id = ?;
SELECT * FROM bid WHERE item_id = ?;

2. SQL層的解決方案

在本方案中,首先通過兩個(gè)步驟對(duì)SQL語句加以改造,從SQL層面上解決這個(gè)問題。

  • 使用關(guān)鍵字UNION ALL將N條查詢語句合為一條語句,便將N+1次查詢轉(zhuǎn)化為了1+1次查詢。
  • 由于第二次查詢的所有記錄被一次性返回,而我們需要將Bid實(shí)體關(guān)聯(lián)到相關(guān)的Item實(shí)體上,因此我們需要添加一個(gè)額外的item_id列以便進(jìn)行實(shí)體關(guān)聯(lián)。

以下是改造后的兩條查詢語句。

SELECT * FROM item;
SELECT ? AS item_id, b.* FROM bid b WHERE item_id = ? UNION ALL
SELECT ? AS item_id, b.* FROM bid b WHERE item_id = ? UNION ALL
SELECT ? AS item_id, b.* FROM bid b WHERE item_id = ? UNION ALL
SELECT ? AS item_id, b.* FROM bid b WHERE item_id = ?;

When we want to query a bid list and every bid entity to carry its item, we can execute two query statements as follows:

SELECT * FROM bid;
SELECT ? AS bid_id, i.* FROM item i WHERE id IN (SELECT item_id FROM bid WHERE id = ?) UNION ALL
SELECT ? AS bid_id, i.* FROM item i WHERE id IN (SELECT item_id FROM bid WHERE id = ?) UNION ALL
SELECT ? AS bid_id, i.* FROM item i WHERE id IN (SELECT item_id FROM bid WHERE id = ?) UNION ALL
SELECT ? AS bid_id, i.* FROM item i WHERE id IN (SELECT item_id FROM bid WHERE id = ?);

ItemBid之間的關(guān)系是典型的一對(duì)多/多對(duì)一關(guān)系。以上這一解決方案也可用于多對(duì)多關(guān)系。

3. ORM應(yīng)用層的解決方案

對(duì)于ORM層,我們需要想辦法從表結(jié)構(gòu)的信息中映射到第二條查詢語句,在Java中開發(fā)中我們常用注解的方式來進(jìn)行配置。

上面的SQL語句中只有四個(gè)要素,兩個(gè)表名itembid,表bid中的外鍵列item_id和表item中的引用列id。 其中,查詢實(shí)體的表名是已知的,于是便只剩下三個(gè)要素。 DoytoQuery定義了一個(gè)注解@DomainPath來配置這三個(gè)要素,用以映射第二條查詢語句。

@Target(FIELD)
@Retention(RUNTIME)
public @interface DomainPath {
    String[] value();
    String localField() default "id";
    String foreignField() default "id";
}

由于第二條查詢語句中附加的id列僅用于實(shí)體賦值,因此我們將附加列的別名統(tǒng)一命名為 MAIN_ENTITY_ID。ItemBid的類定義如下:

@Getter
@Setter
public class ItemView extends AbstractPersistable<Integer> {
    // other fields in Item
    // one-to-many
    // SELECT ? AS MAIN_ENTITY_ID, b.*
    //              FROM bid b           WHERE item_id = ? [UNION ALL ...]
    @DomainPath(value = "bid", foreignField = "item_id")
    private List<BidView> bids;
}
@Getter
@Setter
public class BidView extends AbstractPersistable<Integer> {
    // other fields in Bid
    // many-to-one
    // SELECT ? AS MAIN_ENTITY_ID, i.*
    //              FROM item i           WHERE id      IN (SELECT item_id FROM bid WHERE id = ?)  [UNION ALL ...]
    @DomainPath(value = "item", foreignField = "id", localField = "item_id")
    private ItemView item;
}  

假設(shè)ItemCategory之間的多對(duì)多關(guān)系存放于中間表CATEGORY_ITEM中,我們可以使用@DomainPath來定義如下實(shí)體,用以映射第二條查詢語句:

@Getter
@Setter
public class ItemView extends AbstractPersistable<Integer> {
    // other fields in Item
    // many-to-many
    @DomainPath({"item", "~", "category"})
    private List<ItemView> items;
}
@Getter
@Setter
public class CategoryView extends AbstractPersistable<Integer> {
    // other fields in Category
    // many-to-many
    @DomainPath({"category", "item"})
    private List<ItemView> items;
}

4. 小結(jié)

在本文中,我們介紹了DoytoQuery中的一種可以避免n+1查詢問題的關(guān)聯(lián)查詢方案。并且我們只需要通過一個(gè)注解@DomainPath便可管理ERM中定義的四種實(shí)體關(guān)系,更多關(guān)于DoytoQuery N+1查詢問題的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Spring七大組件是哪些以及作用

    Spring七大組件是哪些以及作用

    這篇文章主要介紹了Spring七大組件是哪些以及作用,幫助剛接觸spring的朋友更快理解,如果有說的不對(duì)的地方還請(qǐng)指正,需要的朋友可以參考下
    2023-03-03
  • 關(guān)于JSON解析中獲取不存在的key問題

    關(guān)于JSON解析中獲取不存在的key問題

    這篇文章主要介紹了關(guān)于JSON解析中獲取不存在的key問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-07-07
  • 基于JAVA中使用Axis發(fā)布/調(diào)用Webservice的方法詳解

    基于JAVA中使用Axis發(fā)布/調(diào)用Webservice的方法詳解

    如果初識(shí)axis發(fā)布/調(diào)用WS,建議先讀上面的參考文件,本文對(duì)于發(fā)布/調(diào)用WS的主要步驟只是簡(jiǎn)單文字描述,沒有它寫的詳盡
    2013-05-05
  • spring boot 集成 swagger3及配置方法

    spring boot 集成 swagger3及配置方法

    Swagger 3是一種開源的API描述工具,它可以幫助開發(fā)人員設(shè)計(jì)、構(gòu)建、文檔化和測(cè)試API,這篇文章主要介紹了spring boot 集成 swagger3,需要的朋友可以參考下
    2023-05-05
  • Java 照片對(duì)比功能的實(shí)現(xiàn)

    Java 照片對(duì)比功能的實(shí)現(xiàn)

    這篇文章主要介紹了Java 照片比對(duì)功能實(shí)現(xiàn)類的示例代碼,幫助大家更好的理解和學(xué)習(xí)Java,感興趣的朋友可以了解下
    2020-12-12
  • Java單例模式實(shí)例簡(jiǎn)述

    Java單例模式實(shí)例簡(jiǎn)述

    這篇文章主要介紹了Java單例模式,在Java應(yīng)用程序設(shè)計(jì)中有著非常重要的作用,本文以實(shí)例形式對(duì)此加以簡(jiǎn)單分析,需要的朋友可以參考下
    2014-09-09
  • Mybatis反向工程出現(xiàn)BigDecimal類型問題及解決

    Mybatis反向工程出現(xiàn)BigDecimal類型問題及解決

    這篇文章主要介紹了Mybatis反向工程出現(xiàn)BigDecimal類型問題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-09-09
  • java實(shí)現(xiàn)多線程之定時(shí)器任務(wù)

    java實(shí)現(xiàn)多線程之定時(shí)器任務(wù)

    本篇文章主要介紹了java實(shí)現(xiàn)多線程之定時(shí)器任務(wù),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-02-02
  • Java Code Cache滿導(dǎo)致應(yīng)用性能降低問題解決

    Java Code Cache滿導(dǎo)致應(yīng)用性能降低問題解決

    這篇文章主要介紹了Java Code Cache滿導(dǎo)致應(yīng)用性能降低問題解決,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-08-08
  • SpringBoot用配置影響B(tài)ean加載@ConditionalOnProperty

    SpringBoot用配置影響B(tài)ean加載@ConditionalOnProperty

    這篇文章主要為大家介紹了SpringBoot用配置影響B(tài)ean加載@ConditionalOnProperty示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-04-04

最新評(píng)論