MyBatis源碼分析(03)SqlSession和Executor

前面的內(nèi)容在源碼中查看了SqlSessionFactory對(duì)象的整個(gè)構(gòu)建過(guò)程子巾,就是下面這一行代碼:

SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

下面我們來(lái)看一下SqlSession對(duì)象的構(gòu)建過(guò)程允趟,也就是下面這一行代碼:

SqlSession session = sqlSessionFactory.openSession();







首先來(lái)看我們的起點(diǎn)逛揩,也就是第一步:

1、SqlSession session = sqlSessionFactory.openSession();

這一步也就是從sqlSessionFactory對(duì)象中獲取SqlSession對(duì)象麸俘,那么是如何獲取的呢辩稽?我們來(lái)打斷點(diǎn),然后繼續(xù)debug:

SqlSessionFactory類是個(gè)接口疾掰,前面大家都知道搂誉,這里面真正創(chuàng)建的對(duì)象是DefaultSqlSessionFactory對(duì)象,那么我們來(lái)進(jìn)入真正實(shí)現(xiàn)的地方:

1静檬、SqlSession session = sqlSessionFactory.openSession();

2炭懊、org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSession()

在這個(gè)方法里面,真正調(diào)用的是openSessionFromDataSource方法拂檩,這個(gè)方法有三個(gè)參數(shù)侮腹,第一個(gè)是獲取默認(rèn)的ExecutorType,這個(gè)是什么呢稻励?進(jìn)入方法看到返回了一個(gè)屬性:

這個(gè)屬性的默認(rèn)值是:

這是從枚舉類里面獲取的SIMPLE值父阻,我們進(jìn)入枚舉類看一下:

這里面定義的三個(gè)常量是我們SQL執(zhí)行器的三個(gè)類型愈涩,分別是簡(jiǎn)單類型,復(fù)用類型加矛,批量類型履婉。如果不指定,那么默認(rèn)就是簡(jiǎn)單類型斟览。這就是openSessionFromDataSource方法的第一個(gè)參數(shù)毁腿,可以看到其他兩個(gè)參數(shù)默認(rèn)是空,下面我們進(jìn)入這個(gè)方法查看:

1苛茂、SqlSession session = sqlSessionFactory.openSession();

2已烤、org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSession()

3、org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSessionFromDataSource

從代碼邏輯可以看出妓羊,進(jìn)入方法后胯究,先拿到全局配置中的環(huán)境配置(configuration.getEnvironment()),大家知道這里面存儲(chǔ)的就是數(shù)據(jù)庫(kù)的連接屬性躁绸,根據(jù)數(shù)據(jù)源的配置裕循,開(kāi)啟數(shù)據(jù)庫(kù)的事務(wù),參數(shù)傳入了事務(wù)等級(jí)和是否自動(dòng)提交净刮,拿到事務(wù)后费韭,根據(jù)事務(wù)和執(zhí)行器類型獲取了一個(gè)執(zhí)行器(Executor),執(zhí)行器是非常重要的一個(gè)環(huán)節(jié)庭瑰,下面看一下如何獲取執(zhí)行器:

1星持、SqlSession session = sqlSessionFactory.openSession();

2、org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSession()

3弹灭、org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSessionFromDataSource

4督暂、org.apache.ibatis.session.Configuration#newExecutor(org.apache.ibatis.transaction.Transaction, org.apache.ibatis.session.ExecutorType)

可以看出前面兩行是空值判斷,下面是根據(jù)三種類型分別創(chuàng)建三種不同的執(zhí)行器穷吮,我們繼續(xù)看逻翁,下面有個(gè)根據(jù)條件?new CachingExecutor 的地方,我們前面說(shuō)過(guò)一級(jí)緩存和二級(jí)緩存捡鱼,這里就是一級(jí)緩存的執(zhí)行器八回,我們繼續(xù)往下看,有這樣一行代碼:

executor = (Executor)interceptorChain.pluginAll(executor);

這樣看代碼作用不太好理解驾诈,但是從單詞意思上看缠诅,調(diào)用的是一個(gè)攔截器鏈方法,沒(méi)錯(cuò)乍迄,這個(gè)方法就是我們mybatis中所有的插件攔截器執(zhí)行的地方管引,我們進(jìn)入這個(gè)方法查看一下邏輯:

1、SqlSession session = sqlSessionFactory.openSession();

2闯两、org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSession()

3褥伴、org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSessionFromDataSource

4谅将、org.apache.ibatis.session.Configuration#newExecutor(org.apache.ibatis.transaction.Transaction, org.apache.ibatis.session.ExecutorType)

5、org.apache.ibatis.plugin.InterceptorChain#pluginAll

這個(gè)方法就非常簡(jiǎn)單了重慢,這里面是所有攔截器的一個(gè)for循環(huán)的執(zhí)行饥臂,我們?cè)谇懊鎸戇^(guò)一個(gè)插件攔截器,即使打印SQL語(yǔ)句輸出執(zhí)行時(shí)間似踱,我們?cè)賮?lái)看一下自定義的攔截器:

這里面也有一個(gè)plugin方法擅笔,攔截器鏈執(zhí)行的就是這個(gè)方法,傳入的參數(shù)就是我們創(chuàng)建的執(zhí)行器屯援,我們?cè)賮?lái)看自定義的攔截器插件類上面的注解,傳入的參數(shù)也有執(zhí)行器的類型念脯,

它會(huì)根據(jù)執(zhí)行器的類型判斷插件是否要執(zhí)行狞洋,這是攔截器起作用的條件。上面就是獲取執(zhí)行器的過(guò)程绿店。我們?cè)賮?lái)回到第三步吉懊,在openSessionFromDataSource方法中,獲取事務(wù)和執(zhí)行器對(duì)象后假勿,就創(chuàng)建了一個(gè)sqlSession對(duì)象:

return new DefaultSqlSession(configuration, executor, autoCommit);

SqlSession是一個(gè)接口借嗽,這里默認(rèn)創(chuàng)建的對(duì)象是DefaultSqlSession對(duì)象,

1转培、SqlSession session = sqlSessionFactory.openSession();

2恶导、org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSession()

3、org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSessionFromDataSource

4浸须、org.apache.ibatis.session.defaults.DefaultSqlSession#DefaultSqlSession(org.apache.ibatis.session.Configuration, org.apache.ibatis.executor.Executor, boolean)

這里只是初始化了幾個(gè)屬性惨寿。DefaultSqlSession類實(shí)現(xiàn)了SqlSession接口,我們來(lái)看下一下SqlSession接口的大致結(jié)構(gòu):

這里面定義了很多像selectOne删窒、selectList裂垦、update等等的操作數(shù)據(jù)庫(kù)的方法,這個(gè)接口繼承了Closeable接口肌索,可以使用TWR形式操作蕉拢。我們?cè)賮?lái)看這個(gè)接口的默認(rèn)實(shí)現(xiàn),DefaultSqlSession類的大致結(jié)構(gòu):

這里面對(duì)基礎(chǔ)的方法有很多默認(rèn)的實(shí)現(xiàn)诚亚,這些實(shí)現(xiàn)其實(shí)就是用類似jdbc的方式訪問(wèn)數(shù)據(jù)庫(kù)晕换,調(diào)用執(zhí)行器去執(zhí)行statement,例如:

執(zhí)行時(shí)加了很多比如緩存的判斷站宗,等等届巩。這些session對(duì)象中默認(rèn)的方法如果直接用來(lái)操作數(shù)據(jù)庫(kù)并不是很友好,意義也不大份乒。最友好的方式是自己定義mapper恕汇。






通過(guò)這個(gè)過(guò)程我們就拿到了SqlSession對(duì)象腕唧。我們?cè)賮?lái)看前面的核心概念,

現(xiàn)在前面四個(gè)對(duì)象的創(chuàng)建過(guò)程我們都了解了瘾英。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末枣接,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子缺谴,更是在濱河造成了極大的恐慌但惶,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,826評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件湿蛔,死亡現(xiàn)場(chǎng)離奇詭異膀曾,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)阳啥,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,968評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門添谊,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人察迟,你說(shuō)我怎么就攤上這事斩狱。” “怎么了扎瓶?”我有些...
    開(kāi)封第一講書人閱讀 164,234評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵所踊,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我概荷,道長(zhǎng)秕岛,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 58,562評(píng)論 1 293
  • 正文 為了忘掉前任误证,我火速辦了婚禮瓣蛀,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘雷厂。我一直安慰自己惋增,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,611評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布改鲫。 她就那樣靜靜地躺著诈皿,像睡著了一般。 火紅的嫁衣襯著肌膚如雪像棘。 梳的紋絲不亂的頭發(fā)上稽亏,一...
    開(kāi)封第一講書人閱讀 51,482評(píng)論 1 302
  • 那天,我揣著相機(jī)與錄音缕题,去河邊找鬼截歉。 笑死,一個(gè)胖子當(dāng)著我的面吹牛烟零,可吹牛的內(nèi)容都是我干的瘪松。 我是一名探鬼主播咸作,決...
    沈念sama閱讀 40,271評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼宵睦!你這毒婦竟也來(lái)了记罚?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 39,166評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤壳嚎,失蹤者是張志新(化名)和其女友劉穎桐智,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體烟馅,經(jīng)...
    沈念sama閱讀 45,608評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡说庭,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,814評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了郑趁。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片刊驴。...
    茶點(diǎn)故事閱讀 39,926評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖穿撮,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情痪欲,我是刑警寧澤悦穿,帶...
    沈念sama閱讀 35,644評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站业踢,受9級(jí)特大地震影響栗柒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜知举,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,249評(píng)論 3 329
  • 文/蒙蒙 一瞬沦、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧雇锡,春花似錦逛钻、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,866評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至立肘,卻和暖如春边坤,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背谅年。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 32,991評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工茧痒, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人融蹂。 一個(gè)月前我還...
    沈念sama閱讀 48,063評(píng)論 3 370
  • 正文 我出身青樓旺订,卻偏偏與公主長(zhǎng)得像弄企,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子耸峭,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,871評(píng)論 2 354