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è)步驟看起來很簡(jiǎn)單,但是用代碼寫出來就很多。我們不妨先記著這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. SQLite教程(六):表達(dá)式詳解2. Mysql入門系列:建立MYSQL客戶機(jī)程序的一般過程3. 導(dǎo)出錯(cuò)誤編碼的mysql數(shù)據(jù)庫(kù)4. SQLite3 API 編程手冊(cè)5. 在Oracle數(shù)據(jù)庫(kù)中移動(dòng)數(shù)據(jù)文件的具體方法6. Navicat for MySQL的使用教程詳解7. 使用 UIMA 和 DB2 Intelligent Miner 進(jìn)行文本挖掘8. SQL語句中的ON DUPLICATE KEY UPDATE使用9. Mysql入門系列:對(duì)MYSQL查詢中有疑問的數(shù)據(jù)進(jìn)行編碼10. mysql啟動(dòng)時(shí)報(bào)錯(cuò) ERROR! Manager of pid-file quit without
