Mybatis Mapper接口工作原理實(shí)例解析
KeyWords: Mybatis 原理,源碼,Mybatis Mapper 接口實(shí)現(xiàn)類,代理模式,動(dòng)態(tài)代理,Java動(dòng)態(tài)代理,
Proxy.newProxyInstance,Mapper 映射,Mapper 實(shí)現(xiàn)
MyBatis 是一款優(yōu)秀的持久層框架,它支持定制化 SQL、存儲(chǔ)過程以及高級(jí)映射。MyBatis 避免了幾乎所有的 JDBC 代碼和手動(dòng)設(shè)置參數(shù)以及獲取結(jié)果集。我們?cè)谑褂?Mybaits 進(jìn)行 ,通常只需要定義幾個(gè) Mapper 接口,然后在編寫一個(gè) xml 文件,我們?cè)谂渲梦募袑懞?sql , Mybatis 幫我們完成 Mapper 接口道具體實(shí)現(xiàn)的調(diào)用。以及將結(jié)果映射到 model bean 中。
我們?cè)陧?xiàng)目中所編寫的眾多的 Mapper 類只是一個(gè)接口(interface ),根據(jù) Java 的多態(tài)性我們知道,可以使用接口接口作為形參,進(jìn)而在運(yùn)行時(shí)確定具體實(shí)現(xiàn)的對(duì)象是什么。但是,對(duì)于 Mapper 接口,我們并沒有編寫其實(shí)現(xiàn)類!Mybatis是如何找到其實(shí)現(xiàn)類,進(jìn)而完成具體的 CRUD 方法調(diào)用的呢?原理何在?
Mapper 接口是怎么找到實(shí)現(xiàn)類的
為了弄清楚 Mapper 接口是如何找到實(shí)現(xiàn)類的,我們先回憶一下 Mybatis 是怎么使用的,根據(jù)實(shí)際的例子,進(jìn)而一點(diǎn)點(diǎn)的去分析。這里的使用指的是Mybatis 單獨(dú)使用,而不是整合 spring , 因?yàn)檎?spring 的話,還需要涉及 Mapper dao 裝載到 spring 容器的問題,spring 幫忙創(chuàng)建數(shù)據(jù)源配置等問題。
通常我們使用 Mybatis 的主要步驟是:
構(gòu)建 SqlSessionFactory ( 通過 xml 配置文件 , 或者直接編寫Java代碼) 從 SqlSessionFactory 中獲取 SqlSession 從SqlSession 中獲取 Mapper 調(diào)用 Mapper 的方法 ,例如:blogMapper.selectBlog(int blogId)從一段代碼看起
上面我們概括了使用 Mybatis 的4個(gè)步驟。這4個(gè)步驟看起來很簡單,但是用代碼寫出來就很多。我們不妨先記著這4個(gè)步驟,再去看代碼,會(huì)容易點(diǎn)。
// 1. DataSource dataSource = BlogDataSourceFactory.getBlogDataSource();TransactionFactory transactionFactory = new JdbcTransactionFactory();Environment environment = new Environment('development', transactionFactory, dataSource);Configuration configuration = new Configuration(environment);configuration.addMapper(BlogMapper.class);// 添加Mapper接口SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(configuration);// 2. SqlSession session = sqlSessionFactory.openSession();try { // 3. BlogMapper mapper = session.getMapper(BlogMapper.class); // 4. Blog blog = mapper.selectBlog(1);} finally { session.close();}
在這塊代碼中,第 1 部分我們使用了 Java 編碼的形式來實(shí)現(xiàn) SqlSessionFactory ,也可以使用 xml 。如果使用xml的話,上面的第一部分代碼就是這樣的:
String resource = 'org/mybatis/example/mybatis-config.xml'; // xml內(nèi)容就不貼了InputStream inputStream = Resources.getResourceAsStream(resource);SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
我們本次的目標(biāo)是弄清楚 “ Mapper 是如何找到實(shí)現(xiàn)類的 ”,我們注意上面代碼 3 , 4 的位置:
// 3. BlogMapper mapper = session.getMapper(BlogMapper.class); // 4. Blog blog = mapper.selectBlog(1);
這里 mapper 可以調(diào)用selectBlog(1) 這個(gè)方法,說明 mapper 是個(gè)對(duì)象,因?yàn)閷?duì)象才具有方法行為實(shí)現(xiàn)啊。BlogMapper接口是不能實(shí)例化的,更沒有具體方法實(shí)現(xiàn)。我們并沒有定義一個(gè)類,讓它實(shí)現(xiàn)BlogMapper接口,而在這里它只是通過調(diào)用session.getMapper() 所得到的。由此,我們可以推斷:肯定是session.getMapper() 方法內(nèi)部產(chǎn)生了BlogMapper的實(shí)現(xiàn)類。有什么技術(shù)可以根據(jù)BlogMapper 接口生成了一個(gè)實(shí)現(xiàn)類呢?想到這里,對(duì)于有動(dòng)態(tài)代理 使用經(jīng)驗(yàn)的程序員來說,很容易想到,這背后肯定是基于動(dòng)態(tài)代理技術(shù),具體怎么實(shí)現(xiàn)的呢?下面我們來根據(jù)源碼一探究竟。
Mapper 接口的注冊(cè)
從上面的代碼中,我們知道 BlogMapper 接口的實(shí)現(xiàn)類是從session.getMapper中得來的,大概是基于動(dòng)態(tài)代理技術(shù)實(shí)現(xiàn)。我們既然能夠從SqlSession中得到BlogMapper接口的,那么我們肯定需要先在哪里把它放進(jìn)去了,然后 SqlSession 才能生成我們想要的代理類啊。上面代碼中有這么一行:
configuration.addMapper(BlogMapper.class);
跟著這個(gè) addMapper 方法的代碼實(shí)現(xiàn)是這樣的:
public <T> void addMapper(Class<T> type) { mapperRegistry.addMapper(type); }
我們看到這里 mapper
相關(guān)文章:
1. MYSQL數(shù)據(jù)庫存文本轉(zhuǎn)存數(shù)據(jù)庫問題2. 學(xué)好Oracle的六條總結(jié)3. MySQL創(chuàng)始人發(fā)郵件尋求中國幫助4. 快速刪除ORACLE重復(fù)記錄5. 巧用SQL語言在ACCESS數(shù)據(jù)庫中批量替換內(nèi)容6. MySQL學(xué)習(xí)記錄之KEY分區(qū)引發(fā)的血案7. ACCESS轉(zhuǎn)SQL數(shù)據(jù)庫相關(guān)的幾個(gè)技能8. Mysql入門系列:MYSQL列類型選擇與MYSQL查詢效率9. 如何將mysql表的內(nèi)容追加導(dǎo)入另一張表10. mysql like語句問題

網(wǎng)公網(wǎng)安備