詳解MyBatis延遲加載是如何實現(xiàn)的
引言
MyBatis 的延遲加載(懶加載)特性允許在需要使用關(guān)聯(lián)對象數(shù)據(jù)時才進(jìn)行加載,而不是在執(zhí)行主查詢時就加載所有相關(guān)數(shù)據(jù)。這種機(jī)制可以提高應(yīng)用程序的性能,特別是當(dāng)關(guān)聯(lián)數(shù)據(jù)龐大或關(guān)聯(lián)層次較深時。我們將通過以下幾個方面來深入了解MyBatis的延遲加載實現(xiàn)機(jī)制。
配置延遲加載
要在MyBatis中啟用延遲加載,需要在配置文件mybatis-config.xml
中進(jìn)行相關(guān)設(shè)置:
<settings> <!-- 開啟全局的延遲加載 --> <setting name="lazyLoadingEnabled" value="true"/> <!-- 將積極加載改為消極加載(即延遲加載) --> <setting name="aggressiveLazyLoading" value="false"/> <!-- 當(dāng)開啟延遲加載時,每個屬性都會延遲加載 --> <setting name="lazyLoadTriggerMethods" value="equals,clone,hashCode,toString"/> </settings>
lazyLoadingEnabled
控制是否啟用延遲加載。aggressiveLazyLoading
控制是否對所有關(guān)聯(lián)對象進(jìn)行加載,設(shè)置為false
可以避免加載不必要的關(guān)聯(lián)對象。lazyLoadTriggerMethods
指定哪些方法觸發(fā)加載屬性,通常保持默認(rèn)即可。
實現(xiàn)原理
MyBatis 的延遲加載主要依賴于代理對象。當(dāng)配置了延遲加載后,MyBatis 會為需要延遲加載的屬性生成一個代理對象,當(dāng)訪問這個屬性時,代理對象負(fù)責(zé)執(zhí)行實際的加載操作。
步驟概述
- 結(jié)果映射: 在結(jié)果映射時,對于配置了延遲加載的關(guān)聯(lián)屬性,MyBatis 不會立即執(zhí)行關(guān)聯(lián)查詢,而是為該屬性創(chuàng)建一個代理對象。
- 代理觸發(fā): 當(dāng)訪問這個屬性的方法時,比如
get
方法,代理對象會觸發(fā)實際的關(guān)聯(lián)查詢。 - 查詢加載: 代理對象執(zhí)行關(guān)聯(lián)查詢,并將查詢結(jié)果設(shè)置到目標(biāo)對象的屬性中,替換掉自身。
代碼演示
以一對多關(guān)系為例,配置延遲加載:
<resultMap id="blogResultMap" type="Blog"> <collection property="posts" ofType="Post" select="selectPostsForBlog" column="id" fetchType="lazy"/> </resultMap> <select id="selectBlog" resultMap="blogResultMap"> SELECT * FROM blog WHERE id = #{id} </select> <select id="selectPostsForBlog" parameterType="int" resultType="Post"> SELECT * FROM post WHERE blog_id = #{id} </select>
這里,<collection>
中的 fetchType="lazy"
指示MyBatis為 posts
屬性創(chuàng)建代理對象,以實現(xiàn)延遲加載。
源碼解析
在MyBatis中,延遲加載的實現(xiàn)涉及到以下幾個關(guān)鍵類:
SqlSessionManager
: 作為SQL會話的管理器,它負(fù)責(zé)創(chuàng)建SqlSession
。Configuration
: 包含了MyBatis的所有配置信息,包括延遲加載的設(shè)置。Executor
: 執(zhí)行器,負(fù)責(zé)執(zhí)行SQL命令,延遲加載的觸發(fā)最終會調(diào)用執(zhí)行器來執(zhí)行關(guān)聯(lián)查詢。ProxyFactory
: 代理工廠,用于創(chuàng)建延遲加載的代理對象。
延遲加載的代理對象主要通過Java的動態(tài)代理實現(xiàn)。在訪問代理對象的方法時,動態(tài)代理會攔截這個調(diào)用,并判斷是否需要觸發(fā)延遲加載。如果需要,則執(zhí)行實際的查詢并加載數(shù)據(jù),然后將結(jié)果設(shè)置到目標(biāo)對象中。
以下是一個簡化的示例來說明代理對象如何攔截方法調(diào)用并觸發(fā)加載:
public class LazyLoadingProxy implements InvocationHandler { private Object target; private boolean loaded; public LazyLoadingProxy(Object target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { if (!loaded && isLoadMethod(method)) { loadObject(); loaded = true; } return method.invoke(target, args); } private boolean isLoadMethod(Method method) { // 判斷方法是否觸發(fā)加載 return "getPosts".equals(method.getName()); } private void loadObject() { // 執(zhí)行加載邏輯,比如執(zhí)行SQL查詢 } }
在上面的代碼中,LazyLoadingProxy
是一個動態(tài)代理類,它在方法調(diào)用時判斷是否需要加載數(shù)據(jù),并在必要時進(jìn)行加載。這個簡化的例子演示了延遲加載的基本思想。
總結(jié)
MyBatis的延遲加載特性通過動態(tài)代理和配置控制,實現(xiàn)了按需加載關(guān)聯(lián)數(shù)據(jù)的能力。通過延遲加載,可以優(yōu)化應(yīng)用程序的性能,特別是在處理復(fù)雜關(guān)系和大量數(shù)據(jù)時。雖然延遲加載增加了實現(xiàn)的復(fù)雜度,但MyBatis通過提供靈活的配置和強(qiáng)大的映射機(jī)制,使得管理這種復(fù)雜度成為可能。
以上就是詳解MyBatis延遲加載是如何實現(xiàn)的的詳細(xì)內(nèi)容,更多關(guān)于MyBatis延遲加載的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
java8升級到j(luò)ava17的兼容性分析與遷移指南
這篇文章主要為大家詳細(xì)介紹了從?Java?8?升級到?Java?17?的詳細(xì)分析和遷移步驟,包括代碼修改建議,依賴更新和配置調(diào)整,有需要的小伙伴可以參考一下2025-04-04解決IntelliJ IDEA 控制臺輸出中文亂碼問題(史上最簡單)
這篇文章主要介紹了史上最簡單的IntelliJ IDEA 控制臺輸出中文亂碼問題的解決方法,非常不錯,具有一定的參考借鑒價值,需要的朋友參考下吧2018-05-05SpringBoot項目啟動后再請求遠(yuǎn)程接口的解決方式
Spring?Boot是由Pivotal團(tuán)隊提供的全新框架,其設(shè)計目的是用來簡化Spring應(yīng)用的創(chuàng)建、運(yùn)行、調(diào)試、部署等,這篇文章主要介紹了SpringBoot項目啟動后再請求遠(yuǎn)程接口的實現(xiàn)方式?,需要的朋友可以參考下2023-02-02Spring WebFlux實現(xiàn)參數(shù)校驗的示例代碼
請求參數(shù)校驗,在實際的應(yīng)用中很常見,網(wǎng)上的文章大部分提供的使用注解的方式做參數(shù)校驗。本文主要介紹 Spring Webflux Function Endpoint 使用 Spring Validation 來校驗請求的參數(shù)。感興趣的可以了解一下2021-08-08Spring boot 跳轉(zhuǎn)到j(luò)sp頁面的實現(xiàn)方法
本篇文章主要介紹了Spring boot 跳轉(zhuǎn)到j(luò)sp頁面的實現(xiàn)方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-04-04Idea安裝及涉及springboot詳細(xì)配置的圖文教程
這篇文章主要介紹了Idea安裝及涉及springboot詳細(xì)配置,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-10-10