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

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

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


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

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