對於Mybatis 擁有的Lazy Load(有中文翻譯成延遲加載)功能,應該很同窗都有據說過,今天主要與你們一塊兒來解讀一下Mybatis在Lazy Load功能的實現的代碼。Lazy Load實現的功能很好理解,就是在數據與對象進行Mapping操做時,只有當真正使用該對象時,才進行Mapping操做,以減小沒必要要的數據庫查詢開銷,從而提高了程序的效率。 sql
首先就從配置部分講起。(本文以Mybatis-3.0.5版本的源代碼進行分析) 數據庫
在配置 SqlSessionFactoryBean 時,須要指定 configLocation 屬性,須要設置 Mybatis Configuration 對象的配置信息,其中有一個配置項目名爲 lazyLoadingEnabled 的設置屬性,就是用來開啓或關閉 Mybatis 的 Lazy Load 功能。默認設置是 false. 能夠看一下 sqlmap-config.xml 文件內容。在 Configuration類的setLazyLoadingEnabled 方法的實現上,還能夠很清楚的分析,Mybatis的lazy load功能是須要藉助Cglib的代理功能來實現的。 apache
接下來,根據以前給你們講Lazy Load的意義時,提供其解決的數據與對象進行Mapping操做時加載優化,那就找到了出現,只要找到Mybatis是如何對數據集與BO對象進行Mapping操做的實現,就應該能夠定位與這個屬性是如何來啓動Lazy Load功能。 session
Mybatis 的Mapping操做都是由 org.apache.ibatis.executor.resultset.ResultSetHandler接口的handleResultSets方法來完成的。並且Mybatis只有一個類實現了這個接口FastResultSetHandler.下面的分析方向很明確了,直接分析一下FastResultSetHandler的handleResultSets方法 app
下面就能夠直接找到實現的代碼重點,FastResultSetHandler 提供一個方法,來實現一行記錄轉成對象的功能。 工具
createResultObject方法 優化
從上面的代碼,能夠很明確的發現 ResultObjectProxy.createProxy 是對BO對象進行的代理實現. 最後只要找到代理的回調實現(Callback),就能夠分析出最終的Lazy Load的實現功能。裏面的分析定位過程就不講了,最終會找到EnhancedResultObjectProxyImpl類。其intercept方法,就是咱們要分析的最終實現的代碼。當BO對象的方法被調用時,就會觸鬚要實施是否進行Lazy Load方式的加載。 spa
lazyLoader.size() 保存須要延遲加載屬性列表的個數。 翻譯
lazyLoader.loadAll 就會觸發ResultLoader的loadResult方法完成數據的加載實現。 代理
至此 Mybatis 的整個 Lazy Load 的功能介紹就到此了。總結一下,其實現的原理就是對 BO 對象,藉助 Cglib 工具,對 BO 對象進行加強。而後在使用 BO 時,進行即時的檢測,來完成數據的加載實現。