對于Mybatis 擁有的Lazy Load(有中文翻譯成延遲加載)功能,應該很同學都有聽說過,今天主要與大家一起來解讀一下Mybatis在Lazy Load功能的實現的代碼。Lazy Load實現的功能很好理解,就是在數據與對象進行Mapping操作時,只有當真正使用該對象時,才進行Mapping操作,以減少不必要的數據庫查詢開銷,從而提升了程序的效率。
首先就從配置部分講起。(本文以Mybatis-3.0.5版本的源代碼進行分析)
在配置SqlSessionFactoryBean時,需要指定configLocation 屬性,需要設置Mybatis Configuration對象的配置信息,其中有一個配置項目名為lazyLoadingEnabled的設置屬性,就是用來開啟或關閉Mybatis的Lazy Load功能。默認設置是 false. 可以看一下 sqlmap-config.xml文件內容。 
Sqlmap-config.xml 文件在SqlSessionFactoryBean 初始化后,解析并加載到 org.apache.ibatis.session.Configuration 該對象上

在 Configuration類的setLazyLoadingEnabled 方法的實現上,還可以很清楚的分析,Mybatis的lazy load功能是需要借助Cglib的代理功能來實現的。

接下來,根據之前給大家講Lazy Load的意義時,提供其解決的數據與對象進行Mapping操作時加載優化,那就找到了出現,只要找到Mybatis是如何對數據集與BO對象進行Mapping操作的實現,就應該可以定位與這個屬性是如何來啟動Lazy Load功能。
Mybatis 的Mapping操作都是由 org.apache.ibatis.executor.resultset.ResultSetHandler接口的handleResultSets方法來完成的。而且Mybatis只有一個類實現了這個接口 FastResultSetHandler.下面的分析方向很明確了,直接分析一下FastResultSetHandler的handleResultSets方法
下面就可以直接找到實現的代碼重點,FastResultSetHandler 提供一個方法,來實現一行記錄轉成對象的功能。


從上面的代碼,可以很明確的發現 ResultObjectProxy.createProxy 是對BO對象進行的代理實現. 最后只要找到代理的回調實現(Callback),就可以分析出最終的Lazy Load的實現功能。里面的分析定位過程就不講了,最終會找到EnhancedResultObjectProxyImpl類。其intercept方法,就是我們要分析的最終實現的代碼。當BO對象的方法被調用時,就會觸需要實施是否進行Lazy Load方式的加載。

lazyLoader.size() 保存需要延遲加載屬性列表的個數。
lazyLoader.loadAll 就會觸發ResultLoader的loadResult方法完成數據的加載實現。
至此Mybatis的整個Lazy Load的功能介紹就到此了。總結一下,其實現的原理就是對BO對象,借助Cglib工具,對BO對象進行增強。然后在使用BO時,進行即時的檢測,來完成數據的加載實現。
Good Luck!
Yours Matthew!