亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

詳解MyBatis日志如何做到兼容所有常用的日志框架

 更新時(shí)間:2020年11月10日 15:00:26   作者:雙子孤狼  
這篇文章主要介紹了詳解MyBatis日志如何做到兼容所有常用的日志框架,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

前言

日志,在我們開(kāi)發(fā)中是一個(gè)非常重要的話題,良好的日志打印可以幫助我們快速的定位問(wèn)題,可能現(xiàn)在我們開(kāi)發(fā)用到最多的日志框架就是slf4j了,但是日志還有其他很多優(yōu)秀的框架,比如:Apache Common Log,Log4j,java.util.logging等。MyBatis作為一款優(yōu)秀的ORM框架,定義了一套統(tǒng)一的日志接口供應(yīng)用層調(diào)用,而底層卻利用適配器模式兼容了我們上面所列出來(lái)的常用日志框架。

MyBatis日志分類

在介紹MyBatis的全局配置文件的時(shí)候,我們提到setting內(nèi)有一個(gè)屬性logImpl,可以配置的選項(xiàng)有:SLF4J,LOG4J,LOG4J2,JDK_LOGGING,COMMONS_LOGGING,STDOUT_LOGGING,NO_LOGGING。
這就說(shuō)明MyBatis支持六種日志類型(NO_LOGGING是不打印日志)。我們看一下MyBatis的日志模塊也可以很明顯的看出六種日志類型:

在這里插入圖片描述

它們的對(duì)應(yīng)關(guān)系為:

日志屬性 對(duì)應(yīng)日志模塊包名 實(shí)現(xiàn)方式
SLF4J slf4j 使用SLF4J日志框架實(shí)現(xiàn)
LOG4J log4j 使用Log4J日志框架實(shí)現(xiàn)(1.x版本)
LOG4J2 log4j2 使用Log4J日志框架實(shí)現(xiàn)(2.x版本)
JDK_LOGGING jdk14 使用java.util.logging實(shí)現(xiàn)
COMMONS_LOGGING commons 使用Apache Commons Logging實(shí)現(xiàn)
STDOUT_LOGGING stdout 使用System類實(shí)現(xiàn)
NO_LOGGING nologging 不打印日志

PS:需要注意的是,SLF4J并不是一個(gè)具體的日志框架,也就是我們不能單獨(dú)只配置SLF4J而不引入其他任何具體的日志框架。

簡(jiǎn)單談?wù)凷LF4J

SLF4J:簡(jiǎn)單日記門面。(英文全稱為simple logging Facade for Java),這個(gè)是用來(lái)為各種日志框架提供一個(gè)簡(jiǎn)單的統(tǒng)一的接口,這樣使得我們?cè)谇袚Q日志框架的時(shí)候可以直接替換jar包就可以了,而無(wú)需修改源代碼。

logback我想大家都用過(guò),logback是一個(gè)實(shí)現(xiàn)了具體日志打印的框架,但是MyBatis上面列出來(lái)的分類并沒(méi)有支持logback,它又為什么能夠打印呢?這就是SLF4J的作用了,因?yàn)閘ogback也實(shí)現(xiàn)了SLF4J提供的接口,所以我們需要將logback和SLF4J結(jié)合配置使用才行。而后面的介紹中也可以看到,MyBatis中如果我們不指定日志種類的時(shí)候,優(yōu)先選擇的就是SLF4J,這正是因?yàn)镾LF4J可以和其他許多日志框架一起結(jié)合來(lái)使用。

那么假如我們指定了日志類型為SLF4J,但是不引入其他任何實(shí)現(xiàn)呢?
答案就是MyBatis不會(huì)打印任何日志出來(lái),下圖就是只配置了SLF4J而沒(méi)有引入其他任何實(shí)現(xiàn)的警告信息:

在這里插入圖片描述

可以看到這里提示我們SLF4J沒(méi)有任何實(shí)現(xiàn),而后面的sql語(yǔ)句和參數(shù)這些信息也沒(méi)有打印出來(lái)。

MyBatis日志實(shí)現(xiàn)原理

日志的解析

老規(guī)矩,我們還是先找到加載mybatis-config配置文件中的解析日志的源碼:

在這里插入圖片描述

在這里插入圖片描述

這里首先會(huì)根據(jù)我們配置的屬性作為別名去TypeAliasRegistry類中查找對(duì)應(yīng)的類,如果不存在這個(gè)別名,那就會(huì)把我們配置的屬性直接通過(guò)Class.forName去查找日志類,所以看到這里就明白我們可以自定義日志類,只要實(shí)現(xiàn)Log接口就行,然后配置我們自己的類名就行了。雖然別名都存在TypeAliasRegistry類里面,但是我們前面介紹MyBatis配置文件的時(shí)候,列出了TypeAliasRegistry類中默認(rèn)初始化的別名,并沒(méi)有看到日志相關(guān)類的別名,那么日志的別名又是在哪里配置的呢?我們打開(kāi)Configuration類:

在這里插入圖片描述

可以看到Configuration的構(gòu)造方法里面也初始化了一些別名注冊(cè)到TypeAliasRegistry類了。
接下來(lái)我們看看讀取到日志類之后調(diào)用了setLogImpl做了什么事情:

在這里插入圖片描述

調(diào)用了LogFactory類的方法。

LogFactory

LogFactory工廠是負(fù)責(zé)創(chuàng)建日志對(duì)象對(duì)應(yīng)的適配器。
LogFactory的靜態(tài)代碼塊內(nèi)按順序初始化了所有內(nèi)置的日志

在這里插入圖片描述

再看一下tryImplementation方法,如果logConstructor不為空,說(shuō)明當(dāng)前還沒(méi)有加載到日志適配器,那就繼續(xù)執(zhí)行run()方法,也就是繼續(xù)執(zhí)行useXXXLogging方法,而所有的useXXXLogging方法都是調(diào)用了setImplementation方法。

在這里插入圖片描述

下面這里如果加載成功之后就會(huì)對(duì)logConstructor進(jìn)行賦值,那么后續(xù)的方法就不會(huì)再執(zhí)行run()方法, 而如果拋出異常,因?yàn)橐呀?jīng)被捕獲了,所以就會(huì)繼續(xù)往后執(zhí)行靜態(tài)代碼塊內(nèi)的方法。

在這里插入圖片描述

從上面的LogFactory中我們可以看到,初始化的時(shí)候就會(huì)默認(rèn)初始化一個(gè)日志適配器,所以如果我們引用了相關(guān)日志所需要的類,那么就會(huì)按照static代碼塊內(nèi)的順序進(jìn)行選擇一個(gè)合適的日志適配器。

繼續(xù)回到上面的Configuration里面,這里拿到我們配置的日志信息之后,會(huì)直接調(diào)用useCustomLogging方法,也就是繞過(guò)了上面的logConstructor == null這個(gè)判斷,而直接調(diào)用了setImplementation方法,所以假如我們配置了日志信息,那么會(huì)覆蓋初始化的日志適配器。

PS:假如我們配置了一個(gè)不存在的日志類,那么因?yàn)樗侵苯诱{(diào)用setImplementation方法,所以異常就會(huì)被拋出來(lái),不拋異常的方法是tryImplementation而不是setImplementation。

jdbc log

MyBatis的日志包下面還有一個(gè)包時(shí)jdbc,這個(gè)我們還沒(méi)有介紹,那么jdbc包下面的類又有什么用呢?我們先看一下類圖關(guān)系:

在這里插入圖片描述

很明顯,MyBatis將日志拆分成了ConnectionLogger,PreparedStatementLogger,ResultSetLogger,StatementLogger四種類型分開(kāi)處理,它們都繼承了BaseJdbcLogger類,而且實(shí)現(xiàn)了InvocationHandler接口,也很明顯,這里用到了JDK動(dòng)態(tài)代理。

任意點(diǎn)開(kāi)ConnectionLogger可以發(fā)現(xiàn),它是用來(lái)代理Connection對(duì)象的:

在這里插入圖片描述

其他三個(gè)那很明顯,分別是用來(lái)代理PreparedStatement,ResultSet,Statement這三個(gè)對(duì)象的。也就是說(shuō)MyBatis中日志最終的打印是通過(guò)JDK動(dòng)態(tài)代理來(lái)實(shí)現(xiàn)的,而且不同的執(zhí)行過(guò)程分成了四個(gè)對(duì)象來(lái)分別負(fù)責(zé)對(duì)應(yīng)的日志打印。

我們繼續(xù)看一下ConnectionLogger的invoke方法,可以看到,這里就是打印了一句日志:

在這里插入圖片描述

上面日志打印出來(lái)的效果就是我們下面紅框中的日志:

在這里插入圖片描述

等等,差點(diǎn)被忽悠了,這代碼里面并沒(méi)有打印“==>”,打印出來(lái)的這個(gè)符號(hào)又是怎么來(lái)的呢?

那就需要進(jìn)入debug方法里面繼續(xù)看一看,這個(gè)debug方法是在抽象類BaseJdbcLogger里面實(shí)現(xiàn)的,所以我們還需要看看BaseJdbcLogger類的debug方法。

在這里插入圖片描述

可以看到這里打印的時(shí)候拼接了一個(gè)前綴:

在這里插入圖片描述

PS:queryStack是查詢層數(shù),如果沒(méi)有嵌套查詢則queryStack=1

總結(jié)

本文主要分析了MyBatis日志的加載原理,并對(duì)LogFactory作為適配器對(duì)象工廠是如何選擇日志適配器對(duì)象進(jìn)行了分析。最后分析了MyBatis是如何通過(guò)動(dòng)態(tài)代理將不同日志類型分為不同對(duì)象來(lái)實(shí)現(xiàn)日志打印功能的。

到此這篇關(guān)于詳解MyBatis日志如何做到兼容所有常用的日志框架的文章就介紹到這了,更多相關(guān)MyBatis兼容所有日志框架內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Spring事務(wù)失效的一種原因關(guān)于this調(diào)用的問(wèn)題

    Spring事務(wù)失效的一種原因關(guān)于this調(diào)用的問(wèn)題

    這篇文章主要介紹了Spring事務(wù)失效的一種原因關(guān)于this調(diào)用的問(wèn)題,本文給大家分享問(wèn)題原因及解決辦法,通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2021-10-10
  • java實(shí)現(xiàn)爬取知乎用戶基本信息

    java實(shí)現(xiàn)爬取知乎用戶基本信息

    這篇文章主要為大家介紹了一個(gè)基于JAVA的知乎爬蟲,抓取知乎用戶基本信息,感興趣的小伙伴們可以參考一下
    2016-05-05
  • Java反射機(jī)制(Reflection)淺析

    Java反射機(jī)制(Reflection)淺析

    這篇文章主要介紹了Java反射機(jī)制(Reflection)淺析,本文以實(shí)例講解Java的反射機(jī)制,需要的朋友可以參考下
    2014-07-07
  • java單例模式學(xué)習(xí)示例

    java單例模式學(xué)習(xí)示例

    java中單例模式是一種常見(jiàn)的設(shè)計(jì)模式,單例模式分三種:懶漢式單例、餓漢式單例、登記式單例三種,下面提供了單例模式的示例
    2014-01-01
  • 微服務(wù)如何通過(guò)feign.RequestInterceptor傳遞參數(shù)

    微服務(wù)如何通過(guò)feign.RequestInterceptor傳遞參數(shù)

    這篇文章主要介紹了微服務(wù)如何通過(guò)feign.RequestInterceptor傳遞參數(shù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • springboot整合freemarker的踩坑及解決

    springboot整合freemarker的踩坑及解決

    這篇文章主要介紹了springboot整合freemarker的踩坑及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-05-05
  • Java實(shí)現(xiàn)郵件發(fā)送功能

    Java實(shí)現(xiàn)郵件發(fā)送功能

    這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)郵件發(fā)送功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • 淺談java object對(duì)象在heap中的結(jié)構(gòu)

    淺談java object對(duì)象在heap中的結(jié)構(gòu)

    本文主要介紹了淺談java object對(duì)象在heap中的結(jié)構(gòu),感興趣的同學(xué),可以參考下。
    2021-06-06
  • Java使用itext5實(shí)現(xiàn)PDF表格文檔導(dǎo)出

    Java使用itext5實(shí)現(xiàn)PDF表格文檔導(dǎo)出

    這篇文章主要介紹了Java使用itext5實(shí)現(xiàn)PDF表格文檔導(dǎo)出,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-01-01
  • Java 獲取當(dāng)前設(shè)備的 IP 地址(最新推薦)

    Java 獲取當(dāng)前設(shè)備的 IP 地址(最新推薦)

    Internet 協(xié)議 (IP) 地址可以是連接到 TCP/IP 網(wǎng)絡(luò)的每個(gè)設(shè)備的標(biāo)識(shí)符,該標(biāo)識(shí)符用于識(shí)別和定位中間通信的節(jié)點(diǎn),這篇文章主要介紹了在 Java 中獲取當(dāng)前設(shè)備的 IP 地址,需要的朋友可以參考下
    2023-06-06

最新評(píng)論