DoytoQuery中的關(guān)聯(lián)查詢方案示例詳解
1. 背景
Java Persistence with Hibernate 在12.2.1小節(jié)使用如下例子描述 n+1查詢問(wèn)題:
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í)行一條額外的查詢語(yǔ)句,當(dāng)item有N條記錄,一共就會(huì)執(zhí)行N+1條查詢語(yǔ)句:
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層的解決方案
在本方案中,首先通過(guò)兩個(gè)步驟對(duì)SQL語(yǔ)句加以改造,從SQL層面上解決這個(gè)問(wèn)題。
使用關(guān)鍵字UNION ALL
將N條查詢語(yǔ)句合為一條語(yǔ)句,便將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)。
以下是改造后的兩條查詢語(yǔ)句。
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 = ?);
Item
和Bid
之間的關(guān)系是典型的一對(duì)多/多對(duì)一關(guān)系。以上這一解決方案也可用于多對(duì)多關(guān)系。
3. ORM應(yīng)用層面的解決方案
對(duì)于ORM層,我們需要想辦法從表結(jié)構(gòu)的信息中映射到第二條查詢語(yǔ)句,Java中常用注解的方式。
上面的SQL語(yǔ)句中只有四個(gè)要素,兩個(gè)表名item
和bid
,表bid
中的外鍵列item_id
和表item
中的引用列id
。 其中,查詢實(shí)體的表名是已知的,于是便只剩下三個(gè)要素。 DoytoQuery定義了一個(gè)注解@DomainPath
來(lái)配置這三個(gè)要素,用以映射第二條查詢語(yǔ)句。
@Target(FIELD) @Retention(RUNTIME) public @interface DomainPath { String[] value(); String localField() default "id"; String foreignField() default "id"; }
由于第二條查詢語(yǔ)句中附加的id
列僅用于實(shí)體賦值,因此我們將附加列的別名統(tǒng)一命名為 MAIN_ENTITY_ID
。Item
和Bid
的類(lèi)定義如下:
@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è)Item
和Category
之間的多對(duì)多關(guān)系存放于中間表CATEGORY_ITEM
中,我們可以使用@DomainPath
來(lái)定義如下實(shí)體,用以映射第二條查詢語(yǔ)句:
@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查詢問(wèn)題的關(guān)聯(lián)查詢方案。并且,我們只需要通過(guò)一個(gè)注解@DomainPath
便可管理ERM中定義的四種實(shí)體關(guān)系。
以上就是DoytoQuery中的關(guān)聯(lián)查詢方案示例詳解的詳細(xì)內(nèi)容,更多關(guān)于DoytoQuery關(guān)聯(lián)查詢方案的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java 執(zhí)行CMD命令或執(zhí)行BAT批處理方式
這篇文章主要介紹了Java 執(zhí)行CMD命令或執(zhí)行BAT批處理方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08java 實(shí)現(xiàn)鏈棧存儲(chǔ)的方法
下面小編就為大家?guī)?lái)一篇java 實(shí)現(xiàn)鏈棧存儲(chǔ)的方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-08-08HTTPClient如何在Springboot中封裝工具類(lèi)
這篇文章主要介紹了HTTPClient如何在Springboot中封裝工具類(lèi)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09Java高效實(shí)現(xiàn)excel轉(zhuǎn)pdf(支持帶圖片的轉(zhuǎn)換)
這篇文章主要為大家詳細(xì)介紹了如何用java實(shí)現(xiàn)excel轉(zhuǎn)pdf文件,并且支持excel單元格中帶有圖片的轉(zhuǎn)換,文中的示例代碼講解詳細(xì),需要的可以參考下2024-01-01java高效打印一個(gè)二維數(shù)組的實(shí)例(不用遞歸,不用兩個(gè)for循環(huán))
下面小編就為大家?guī)?lái)一篇java高效打印一個(gè)二維數(shù)組的實(shí)例(不用遞歸,不用兩個(gè)for循環(huán))。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-03-03SpringBoot使用布隆過(guò)濾器解決緩存穿透問(wèn)題
緩存穿透是指當(dāng)緩存系統(tǒng)中無(wú)法命中需要的數(shù)據(jù)時(shí),會(huì)直接請(qǐng)求底層存儲(chǔ)系統(tǒng)(如數(shù)據(jù)庫(kù)),但是如果請(qǐng)求的數(shù)據(jù)根本不存在,那么大量的請(qǐng)求就會(huì)直接穿透緩存層,本文將給大家介紹一下SpringBoot使用布隆過(guò)濾器解決緩存穿透問(wèn)題,需要的朋友可以參考下2023-10-10MyBatis查詢數(shù)據(jù),賦值給List集合時(shí),數(shù)據(jù)缺少的問(wèn)題及解決
這篇文章主要介紹了MyBatis查詢數(shù)據(jù),賦值給List集合時(shí),數(shù)據(jù)缺少的問(wèn)題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01