mybatis集成到spring的方式有哪些
本文講解"mybatis集成到spring的方式有哪些",希望能夠解決相關(guān)問題。
1 前言
1.1 集成spring前使用mybatis的方式
mybatis單獨(dú)使用時(shí),一般的寫法如下所示:
// mybatis初始化 String resource = "mybatis-config.xml"; InputStream inputStream = Resources.getResourceAsStream(resource); // 讀取配置文件,創(chuàng)建SqlSessionFactory SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream); // 打開SqlSession SqlSession sqlSession = sqlSessionFactory.openSession(); // 執(zhí)行SQL語(yǔ)句 List list = sqlSession.selectList("com.gameloft9.demo.dataaccess.dao.system.UserMapper.getByLoginName");
1.2 集成mybatis到spring的關(guān)鍵步驟
將mybatis集成到spring之后,就可以被spring的ioc容器托管,再也不用自己創(chuàng)建SqlSessionFactory 、打開SqlSession等操作。具體的集成方法可以參考之前寫的文章:spring集成mybatis進(jìn)行數(shù)據(jù)庫(kù)訪問,其中最重要的配置就是定義好SqlSessionFactoryBean,如下所示:
<!--mybatis?sqlSeesionFactory配置--> <bean id="sqlsessionfactory" class="org.mybatis.spring.sqlsessionfactorybean"> <property name="datasource" ref="datasource"> <property name="configlocation" value="classpath:mybatis-config.xml"> <property?name="mapperlocations"?value="classpath:mapper *mapper.xml"?=""> <property name="typealiasespackage" value="com.gameloft9.demo.dataaccess.model"> </property></property?name="mapperlocations"?value="classpath:mapper></property></property></bean>
因此想要了解spring集成mybatis的原理,就應(yīng)該從SqlSessionFactoryBean入手。
2 SqlSessionFactoryBean對(duì)象分析
SqlSessionFactoryBean,顧名思義跟SqlSessionFactory有著莫大的關(guān)系,它的類結(jié)構(gòu)如下所示:
SqlSessionFactoryBean實(shí)現(xiàn)了InitializingBean接口,我們知道InitializingBean在Bean的生命周期里面扮演了重要的角色,spring創(chuàng)建bean的流程大概是:
1、創(chuàng)建beanFactory
2、加載beanDefinition
3、通過反射創(chuàng)建bean實(shí)例
4、bean的生命周期擴(kuò)展點(diǎn)調(diào)用
其中第4步驟又包含:
5、如果有任何與bean相關(guān)聯(lián)的BeanPostProcessors,Spring會(huì)在postProcesserBeforeInitialization()方法內(nèi)調(diào)用它們。
6、如果bean實(shí)現(xiàn)IntializingBean了,調(diào)用它的afterPropertySet方法。
7、如果bean定義了init-method,調(diào)用init方法
8、如果有BeanPostProcessors 和bean 關(guān)聯(lián),這些bean的postProcessAfterInitialization() 方法將被調(diào)用。
9、如果bean實(shí)現(xiàn)了 DisposableBean,它將調(diào)用destroy()方法。
10、如果bean定義了destroy-method,調(diào)用destroy方法。
因此在我們xml配置好的dataSource,configLocation等屬性設(shè)置好后,SqlSessionFactoryBean就在afterPropertySet()方法里面對(duì)SqlSessionFactory進(jìn)行初始化。
2.1 buildSqlSessionFactory做了什么事情?
buildSqlSessionFactory里面就是具體怎么創(chuàng)建SqlSeesionFactory的,代碼流程比較長(zhǎng),我們用一個(gè)簡(jiǎn)單時(shí)序圖來展示:
在解析完各種配置后,調(diào)用了return this.sqlSessionFactoryBuilder.build(configuration);來創(chuàng)建buildSqlSessionFactory,是不是和之前的方式很類似了?只是一個(gè)是入?yún)⑹莚esource,一個(gè)是我們解析后的配置configuration對(duì)象。
2.2 為什么是SqlSessionFactoryBean卻可以使用SqlSessionFactory?
我們注冊(cè)的是SqlSessionFactoryBean這個(gè)bean,為什么卻說SqlSessionFactory也成為了spring的bean呢?因?yàn)槲覀兊腟qlSessionFactoryBean還實(shí)現(xiàn)了FactoryBean這個(gè)接口。
Spring 中有兩種類型的Bean,一種是普通Bean,另一種是工廠Bean 即 FactoryBean。FactoryBean跟普通Bean不同,其返回的對(duì)象不是指定類的一個(gè)實(shí)例,而是該FactoryBean的getObject方法所返回的對(duì)象。
一般情況下,Spring通過反射機(jī)制利用的class屬性指定實(shí)現(xiàn)類實(shí)例化Bean,在某些情況下,實(shí)例化Bean過程比較復(fù)雜,Spring為此提供了一個(gè)org.springframework.bean.factory.FactoryBean的工廠類接口,用戶可以通過實(shí)現(xiàn)該接口定制實(shí)例化Bean的邏輯。
然后回到maybatis這里,我們通過時(shí)序圖確實(shí)發(fā)現(xiàn)創(chuàng)建SqlSessionFactory十分復(fù)雜,要解析大量的配置之后才能創(chuàng)建,因此我們實(shí)現(xiàn)了FactoryBean接口:
在getObject里面選擇性的進(jìn)行初始化并返回SqlSessionFactory對(duì)象。
至于獲取SqlSessionFactorybean實(shí)例的時(shí)候,是怎么走到FactoryBean的getObject的方法的,可以參考AbstractBeanFactory.getBean方法,這屬于spring ioc范疇了,這個(gè)需要很長(zhǎng)的篇幅才能講清楚,有機(jī)會(huì)我會(huì)單獨(dú)寫一篇spring怎么創(chuàng)建bean和獲取bean的。
3 驗(yàn)證demo
如果你有demo的話,可以單步跟蹤。如果沒有的話,可以使用作者之前做過的一個(gè)基于layui的通用后臺(tái)系統(tǒng),簡(jiǎn)單的安裝mysql和客戶端之后(本地安裝mysql和客戶端真的很簡(jiǎn)單30分鐘搞定吧)、稍微改下配置就可以直接運(yùn)行這個(gè)項(xiàng)目(這個(gè)項(xiàng)目麻雀雖小,但五臟俱全,如果覺得好用,可以幫忙點(diǎn)個(gè)start)。
驗(yàn)證之前,我們先做個(gè)猜想:SqlSessionFactory是在afterPropertiesSet里初始化的還是在getObject里判斷為空再初始化的?
我們分別在SqlSessionFactoryBean的afterPropertiesSet和getObject打上斷點(diǎn),然后debug項(xiàng)目,如下所示:
可以看到,在SqlSessionFactoryBean實(shí)例創(chuàng)建后,在afterPropertiesSet對(duì)SqlSessionFactory進(jìn)行了初始化和創(chuàng)建。然后在獲取它的時(shí)候,走的是getObject方法(這時(shí)候sqlSessionFactory已經(jīng)不為空了):
4 舉一反三
4.1 插件的配置是在SqlSessionFactoryBean的定義里面還是configuration的xml結(jié)點(diǎn)下面?
demo里的插件我們是寫在mybatis-config.xml里面的configuration結(jié)點(diǎn)里面的,如下所示:
這是通用的做法,當(dāng)然你也可以經(jīng)??吹接腥伺渲迷赟qlSessionFactoryBean的定義里面,如下所示:
那么問題來了,這兩個(gè)有什么區(qū)別嗎?
這個(gè)比較簡(jiǎn)單,作者自問自答一下。
這兩個(gè)都可以作為插件的配置方式,都會(huì)起作用,因?yàn)樵赽uildSqlSessionFactory過程中,如果發(fā)現(xiàn)插件不為空,就會(huì)添加到插件集合里面去,如下所示:
然后再解析xml時(shí),又會(huì)將configuration的配置解析一遍,里面也包含插件的解析過程:
所以我們只需要在一個(gè)地方配置好就行了。然后它們有一個(gè)細(xì)微的差別,因?yàn)閏onfiguration是mybatis的配置,所以在configuration中的插件配置,是通過調(diào)用setProperties方法將屬性傳遞進(jìn)來,然后設(shè)置到插件的屬性中。SqlSessionFactoryBean的插件配置不會(huì)調(diào)用setProperties方法,直接走的setter注入將屬性注入進(jìn)去。
這個(gè)大家也可以自行驗(yàn)證下。
4.2 mybatis的事務(wù)管理和spring的事務(wù)管理什么關(guān)系?
如果你仔細(xì)看了SqlSessionFactoryBean創(chuàng)建SqlSessionFactory的過程,那么肯定會(huì)注意到有這么一個(gè)調(diào)用:
if (this.transactionFactory == null) { this.transactionFactory = new SpringManagedTransactionFactory(); }
因?yàn)槲覀兣渲玫氖莝pring的事務(wù),在mybatis的配置文件里面沒有為SqlSessionFactoryBean配置事務(wù),如下所示:
所以那段代碼里面transactionFactory會(huì)因?yàn)?=null,而創(chuàng)建一個(gè)SpringManagedTransactionFactory,下面的截圖也剛好印證了我們的觀點(diǎn)。
那么問題來了,mybatis的事務(wù)和spring的事務(wù)是什么關(guān)系?它們是怎么配合的?
這個(gè)問題回答比較復(fù)雜,需要有spring事務(wù)的知識(shí),大家可以嘗試自己找答案,或者等作者后續(xù)的文章來解答。
5 總結(jié)
mybatis通過SqlSessionFactoryBean將SqlSessionFactory對(duì)象集成到spring中,它實(shí)現(xiàn)了InitializingBean接口,在SqlSessionFactoryBean初始化時(shí)解析配置并創(chuàng)建DefaultSqlSessionFactory對(duì)象。它還實(shí)現(xiàn)了FactoryBean,在getObject時(shí)返回我們創(chuàng)建好的DefaultSqlSessionFactory,使得DefaultSqlSessionFactory也被spring管理起來。
很多框架集成到spring的方法基本都是靠InitializingBean和FactoryBean這兩個(gè)接口來實(shí)現(xiàn)的,這是一種非常好的設(shè)計(jì),值得我們好好學(xué)習(xí)。
- 一文解析spring中事務(wù)的傳播機(jī)制
- mybatis集成到spring的方式有哪些
- Java讀取resources中資源文件路徑以及jar中文件無法讀取如何解決
- MyBatis批量查詢、插入、更新、刪除如何實(shí)現(xiàn)
- java只返回實(shí)體類中的部分字段問題如何解決
- JavaScrip簡(jiǎn)單數(shù)據(jù)類型隱式轉(zhuǎn)換如何實(shí)現(xiàn)
- javaweb中servlet技術(shù)的用法
- Springboot自帶線程池怎么實(shí)現(xiàn)
- java的預(yù)定義Class對(duì)象的方法
- 解決Springboot全局異常處理與AOP日志處理中@AfterThrowing失效問題的方法
- Java使用HttpUtils怎么實(shí)現(xiàn)發(fā)送HTTP請(qǐng)求
- 關(guān)于Spring的統(tǒng)一功能處理(攔截器)實(shí)現(xiàn)
- Hadoop?MapReduce實(shí)現(xiàn)單詞計(jì)數(shù)(Word?Count)
- 詳解SpringBoot可執(zhí)行Jar包運(yùn)行原理
- 詳解Java中如何定義和訪問靜態(tài)字段和方法
- Java通過Lambda表達(dá)式實(shí)現(xiàn)簡(jiǎn)化代碼
- 使用IDEA反編譯沒有擦除泛型的原因解析
- java解決動(dòng)態(tài)配置字段需求問題介紹
- IDEA創(chuàng)建SpringBoot父子Module項(xiàng)目的實(shí)現(xiàn)方法
- Java中局部變量和成員變量有什么區(qū)別