iBATIS DAO框架淺析
iBATIS DAO框架的書寫方便,本文采用如下簡寫約定:
◆Transaction:Tx
◆Manager:Mgr
◆Context:Ctx
◆Interface:Iface
iBATIS DAO框架如圖:
iBATIS DAO的核心在于DaoManager,DaoManager的創(chuàng)建代碼如下:
- Reader reader = Resources.getResourceAsReader("dao.xml");
- DaoManager daoMngr = DaoManagerBuilder.buildDaoManager(reader);
DaoManager是接口,查看DaoManagerBuilder源代碼可發(fā)現(xiàn),其buildDaoManager方法返回的是一個StandardDaoManager實例。buildDaoManager方法調用了XmlDaoManagerBuilder類的buildDaoManager方法,該方法完成如下工作:
1.創(chuàng)建一個StandardDaoManager實例stdDaoMgr;
2.創(chuàng)建一個用于全局收集各種property(來自<properties>元素指向的資源文件或來自當前dao.xml中的各級<property>元素)的Properties對象;
3.解析dao.xml文件(建議閱讀本文時參考一份dao.xml文件,如JGameStore應用中給出的dao.xml)中的<properties>元素,將相應property加入;
4.解析dao.xml文件中的<context>元素,得到一個DaoContext實例daoCtx(4.1);并將調用stdDaoMgr.addContext方法將daoCtx添加到stdDaoMgr中(4.2):
4.1 解析dao.xml文件的<context>元素得到daoCtx的過程為:
實例化一個DaoContext對象daoCtx;
將其daoManager字段設為我們的stdDaoMgr;
如<context>有id屬性,則將daoCtx的id字段取為此屬性的值;
解析<context>的子元素:
4.1.1解析<txMgr>子元素,得到DaoTxMgr接口實例txMgr,設為daoCtx的相應字段,解析過程為:
根據(jù)<txMgr>子元素的type屬性,實例化一個相應的DaoTxMgr實例txMgr;
解析<txMgr>的<property>子元素,將所得property添加入properties;
根據(jù)properties對txMgr進行配置(即調用txMgr.configure方法);
4.1.2解析<dao>子元素,得到一個DaoImpl類實例daoImpl,然后將其加入daoCtx;
4.1.2.1解析過程為:
4.1.2.1.1實例化一個DaoImpl類實例daoImpl;
4.1.2.1.2將daoImpl的daoMgr字段設為我們的stdDaoMgr;
4.1.2.1.3將daoImpl的daoCtx字段設為我們的daoCtx;
4.1.2.1.4將daoImpl的daoIface字段設為<dao>的iface屬性值對應的class;
4.1.2.1.5將daoImpl的daoImplementation字段設為<dao>的implementation屬性值對應的class;
4.1.2.1.6根據(jù)implementation屬性實例化一個DAO實現(xiàn)類,設為daoInstance字段值,注意,該實例一定是一個Dao接口實例,因為任何一個都繼承自DaoTemplate,而DaoTemplate實現(xiàn)了Dao接口;
4.1.2.1.7創(chuàng)建一個當前DAO實現(xiàn)類的代理,設為daoImpl的proxy字段值,該代理在啟用顯式事務時會在調用委托方法前調用daoCtx.startTx方法;在使用隱式事務時則在調用委托方法的前后分別調用daoCtx.startTx方法和commitTx方法(在finally塊中還調用daoCtx.endTx方法)。
4.1.2.2將daoImpl加入daoCtx的過程為:以當前daoImpl填充一張從daoIface到DaoImpl實例的表;
4.2 調用stdDaoMgr.addContext方法將daoCtx添加到stdDaoMgr中的過程為:
4.2.1以當前daoCtx填充一張由id到DaoCtx實例的表;
4.2.2遍歷daoCtx中存放的所有daoImpl,填充一張從daoIface到daoCtx的表和一張從Dao接口實例(即daoImpl中的proxy和daoInstance)到daoCtx的表;
5.客戶以某DaoIface調用DaoMgr.getDao方法得到一個DaoIface實現(xiàn)類實例xxxYyyDao的過程為:
stdDaoMgr查找其從daoIface到daoCtx的表,得到當前daoIface所在daoCtx,然后調用daoCtx.getDao方法:
daoCtx查找其從daoIface到DaoImpl實例的表,得到daoImpl,返回其proxy字段;
6.隱式事務:
隱式事務中,客戶每調用一個xxxYyyDao中方法時,都是一次完整的事務,因為xxxYyyDao是調用DaoMgr.getDao方法得到的,而根據(jù)5,其實xxxYyyDao是一個代理,又根據(jù)4.1.2.1.7,該代理會“在調用其委托方法前后分別調用daoCtx.startTx方法和commitTx方法(在finally塊中還調用daoCtx.endTx方法)”。
6.1 daoCtx.startTx方法調用其txMgr字段的txMgr.startTx方法,該方法返回一個DaoTx實例daoTx,daoCtx將它放入一個線程變量中;
6.2 DaoIface實現(xiàn)類中,由于其一定繼承自某個DaoTemplate,以調用其中的數(shù)據(jù)庫訪問方法,而這些數(shù)據(jù)庫訪問方法都會以自己作為參數(shù)調用daoMgr的getTx方法;該方法查找4.2.2中提到的從Dao接口實例到daoCtx的表,得到一個daoCtx,然后調用daoCtx.getTx;daoCtx.getTx將存儲在線程變量中的daoTx實例返回;
6.3 daoTx實例包含數(shù)據(jù)庫操作所需的關鍵元素,例如對于SqlMapDaoTx,其中就包含一個SqlMapClient實例,SqlMapDaoTemplate中的數(shù)據(jù)庫訪問方法(如insert,queryForList等)都是先調用daoMgr.getTx,得到daoTx實例,將其強制轉化為SqlMapDaoTx實例,然后調用其getSqlMap方法得到SqlMapClient實例,再調用SqlMapClient實例中的相應方法;又如對于JDBC的情況,對應DaoTx為ConnectionDaoTx,該類包含一個,每次調用JdbcDaoTemplate方法的getConnection方法時,該方法都先調用daoMgr.getTx,得到daoTx實例,將其強制轉化為ConnectionDaoTx實例,然后調用其getConnection方法得到其中的Connection實例,然后調用其中的相應方法。
6.4 daoCtx.commitTx方法調用其txMgr字段的txMgr.commitTx(daoTx)方法完成事務的提交。
6.5 daoCtx.endTx方法調用其txMgr字段的txMgr.endTx(daoTx)方法結束事務。
7.顯式事務:
顯式事務通常包括三個步驟:首先,調用daoMgr.startTx,然后調用xxxYyyDao中的方法,***調用daoMgr.commitTx。
7.1 daoMgr.startTx的工作非常簡單,只是設置stdDaoMgr中標記顯式事務的字段;
7.2 調用xxxYyyDao中的方法時,由于代理,將先調用daoCtx.startTx,此過程同6.1;
7.3 調用daoMgr.commitTx時,該方法最終調用的也是daoCtx.commitTx,請參考6.4
下面以一個問題的實現(xiàn)來完成本文的總結工作:如果要由我來實現(xiàn)iBATIS的DAO框架對于Hibernate的支持,我們應該如何實現(xiàn)?
Hibernate的核心在于Session,所有的數(shù)據(jù)庫操作都可調用Session上的相應方法完成,所有考慮用于支持Hibernate的DaoTx實現(xiàn)應該是對Session的一個包裝,該實現(xiàn)中有一個返回當前Session的getSession方法(當然也包括提交和回滾方法)。同樣的,DaoTxMgr實現(xiàn)類的configure方法負責完成某個Session實例(session)的配置,startTx方法負責返回一個包裝了當前session實例的DaoTx實例,commitTx方法將傳入的daoTx實例強制轉化后調用daoTx上的commit方法,rollbackTx方法將傳入的daoTx實例強制轉化后調用daoTx上的rollback方法。而HibernateDaoTemplate類的關鍵就在于其protected的getSession方法,該方法先調用daoMgr.getTx得到當前daoTx實例,強制轉化后調用daoTx上的getSession方法即可。
查詢iBATIS的源代碼,發(fā)現(xiàn)與以上思路完全相同。
iBATIS DAO框架就向你介紹到這里,希望對你了解iBATIS DAO框架有所幫助。
【編輯推薦】