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的類定義如下:
@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)文章
LoggingEventAsyncDisruptorAppender類執(zhí)行流程源碼解讀
這篇文章主要介紹了LoggingEventAsyncDisruptorAppender類執(zhí)行流程源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12
Java使用elasticsearch基礎(chǔ)API使用案例講解
這篇文章主要介紹了Java使用elasticsearch基礎(chǔ)API使用案例講解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08
springboot如何去獲取前端傳遞的參數(shù)的實(shí)現(xiàn)
這篇文章主要介紹了springboot如何去獲取前端傳遞的參數(shù)的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
利用Java實(shí)現(xiàn)調(diào)用http請(qǐng)求
在實(shí)際開(kāi)發(fā)過(guò)程中,我們經(jīng)常需要調(diào)用對(duì)方提供的接口或測(cè)試自己寫的接口是否合適。本文就為大家準(zhǔn)備了幾個(gè)java調(diào)用http請(qǐng)求的幾種常見(jiàn)方式,需要的可以參考一下2022-08-08
Java Thread多線程開(kāi)發(fā)中Object類詳細(xì)講解
這篇文章主要介紹了Java Thread多線程開(kāi)發(fā)中Object類,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧2023-03-03
Java實(shí)現(xiàn)批量發(fā)送帶附件的郵件代碼
大家好,本篇文章主要講的是Java實(shí)現(xiàn)批量發(fā)送帶附件的郵件代碼,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽2022-01-01
SpringCloud使用CircuitBreaker實(shí)現(xiàn)熔斷器的詳細(xì)步驟
在微服務(wù)架構(gòu)中,服務(wù)之間的依賴調(diào)用非常頻繁,當(dāng)一個(gè)下游服務(wù)因高負(fù)載或故障導(dǎo)致響應(yīng)變慢或不可用時(shí),可能會(huì)引發(fā)上游服務(wù)的級(jí)聯(lián)故障,最終導(dǎo)致整個(gè)系統(tǒng)崩潰,熔斷器是解決這類問(wèn)題的關(guān)鍵模式之一,Spring Cloud提供了對(duì)熔斷器的支持,本文將詳細(xì)介紹如何集成和使用它2025-02-02
java 中類似js encodeURIComponent 函數(shù)的實(shí)現(xiàn)案例
這篇文章主要介紹了java 中類似js encodeURIComponent 函數(shù)的實(shí)現(xiàn)案例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-10-10

