欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

MyBatis是如何實(shí)現(xiàn)日志模塊的詳解

 更新時(shí)間:2019年10月24日 08:29:02   作者:成長(zhǎng)的小樹  
這篇文章主要給大家介紹了關(guān)于MyBatis是如何實(shí)現(xiàn)日志模塊的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者使用MyBatis具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

場(chǎng)景復(fù)現(xiàn)

你知道MyBatis是怎么實(shí)現(xiàn)日志的?額,這個(gè)簡(jiǎn)單,我知道啊!不就是在mybatis-config.xml文件中配置一下嗎?

<!DOCTYPE configuration
  PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
  <settings>
  	<setting name="logImpl" value="SLF4J"/>
  </settings>
</configuration>

看,就是上面這么配置就行了,這樣還可以實(shí)現(xiàn)日志切換呢!

看官方文檔介紹,有如下日志實(shí)現(xiàn)可以切換:

你知道具體的原理嗎?MyBatis是怎樣實(shí)現(xiàn)日志切換的呢?

額,這個(gè),這個(gè)...(撓頭,源碼還沒(méi)看過(guò)呢,😓)

源碼擼起來(lái)~

對(duì)于這個(gè)不確定的事,作為程序員的我,怎能輕易說(shuō)不知道呢?!源碼走起來(lái)~

找到一個(gè)入口

首先,先從GitHub上將MyBatis的源碼給clone到本地.老大們擼了那么多行代碼,從哪開始下手呢???

大佬們,也是很規(guī)范的,寫完代碼有不少的Junit測(cè)試類,從test/java中找到了logging包.那就從這開始吧~

@Test
public void shouldReadLogImplFromSettings() throws Exception {
//try-with-resources語(yǔ)句,目的就是為了讀取配置文件IO
//mybatis-config.xml配置文件內(nèi)容如上,<setting name="logImpl" value="SLF4J"/>
try (Reader reader = Resources.getResourceAsReader("org/apache/ibatis/logging/mybatis-config.xml")) {
 //解析配置文件,解析配置文件的代碼就在這了~
 new SqlSessionFactoryBuilder().build(reader);
}

Log log = LogFactory.getLog(Object.class);
log.debug("Debug message.");
assertEquals(log.getClass().getName(), NoLoggingImpl.class.getName());
}

SqlSessionFactoryBuilder類

//根據(jù)配置文件IO構(gòu)建SqlSessionFactory
public SqlSessionFactory build(Reader reader) {
  return build(reader, null, null);
}


public SqlSessionFactory build(Reader reader, String environment, Properties properties) {
  try {
    //構(gòu)建XML配置構(gòu)建器
    XMLConfigBuilder parser = new XMLConfigBuilder(reader, environment, properties);
    //解析XML配置文件
    return build(parser.parse());
  } catch (Exception e) {
    throw ExceptionFactory.wrapException("Error building SqlSession.", e);
  } finally {
    ErrorContext.instance().reset();
    try {
      reader.close();
    } catch (IOException e) {
      // Intentionally ignore. Prefer previous error.
    }
  }
}

XMLConfigBuilder XML配置構(gòu)建器

構(gòu)造函數(shù)

//reader是XML配置文件IO
public XMLConfigBuilder(Reader reader, String environment, Properties props) {
  this(new XPathParser(reader, true, props, new XMLMapperEntityResolver()), environment, props);
}

//實(shí)際調(diào)用的XML配置文件構(gòu)建器構(gòu)建函數(shù)
private XMLConfigBuilder(XPathParser parser, String environment, Properties props) {
  //注意了,這是重點(diǎn),在這地方調(diào)用了父類的構(gòu)造函數(shù),新建了一個(gè)Configuration對(duì)象,下面看看Configuration構(gòu)造函數(shù)做了什么
  super(new Configuration());
  ErrorContext.instance().resource("SQL Mapper Configuration");
  this.configuration.setVariables(props);
  this.parsed = false;
  this.environment = environment;
  this.parser = parser;
}

parse解析方法

/**
 * 解析 mybatis-config.xml
 * @return
 */
public Configuration parse() {
  // 只能解析一次
  if (parsed) {
    throw new BuilderException("Each XMLConfigBuilder can only be used once.");
  }
  parsed = true;
  //根據(jù)XPATH獲取configuration節(jié)點(diǎn)
  parseConfiguration(parser.evalNode("/configuration"));
  return configuration;
}

parseConfiguration 解析配置方法

private void parseConfiguration(XNode root) {
  try {
    //issue #117 read properties first
    // 解析 properties 節(jié)點(diǎn)
    propertiesElement(root.evalNode("properties"));
    // 解析 settings 節(jié)點(diǎn)
    Properties settings = settingsAsProperties(root.evalNode("settings"));
    loadCustomVfs(settings);
    // 解析 typeAliases 節(jié)點(diǎn)
    typeAliasesElement(root.evalNode("typeAliases"));
    // 解析 plugins 節(jié)點(diǎn)
    pluginElement(root.evalNode("plugins"));
    // 解析 objectFactory 節(jié)點(diǎn)
    objectFactoryElement(root.evalNode("objectFactory"));
    // 解析 objectWrapperFactory 節(jié)點(diǎn)
    objectWrapperFactoryElement(root.evalNode("objectWrapperFactory"));
    // 解析 reflectorFactory 節(jié)點(diǎn)
    reflectorFactoryElement(root.evalNode("reflectorFactory"));
    settingsElement(settings);
    // 解析 environments 節(jié)點(diǎn), 需要在 objectFactory 和 objectWrapperFactory才能讀取
    environmentsElement(root.evalNode("environments"));
    // 解析 databaseIdProvider 節(jié)點(diǎn)
    databaseIdProviderElement(root.evalNode("databaseIdProvider"));
    // 解析 typeHandlers 節(jié)點(diǎn)
    typeHandlerElement(root.evalNode("typeHandlers"));
    // 解析 mappers 節(jié)點(diǎn)
    mapperElement(root.evalNode("mappers"));
  } catch (Exception e) {
    throw new BuilderException("Error parsing SQL Mapper Configuration. Cause: " + e, e);
  }
}

其中Properties settings = settingsAsProperties(root.evalNode("settings"));是解析setttings節(jié)點(diǎn),將settings節(jié)點(diǎn)的內(nèi)容解析到 Properties 對(duì)象,其中涉及到很多操作,不在本文分析之列.

settingsElement(settings);將settings中的配置設(shè)置到Configuration對(duì)象.

settingsElement settings節(jié)點(diǎn)的配置

private void settingsElement(Properties props) throws Exception {
  ...
  configuration.setLogPrefix(props.getProperty("logPrefix"));
  @SuppressWarnings("unchecked")
  // 這個(gè)不就是從settings中解析日志的實(shí)現(xiàn)嗎?快看看~
  // resovleClass是父類BaseBuilder中的方法
  Class<? extends Log> logImpl = (Class<? extends Log>) resolveClass(props.getProperty("logImpl"));
  //將日志的實(shí)現(xiàn)類設(shè)置到configuration配置類中
  configuration.setLogImpl(logImpl);
  configuration.setConfigurationFactory(resolveClass(props.getProperty("configurationFactory")));
}

Configuration 配置類

public class Configuration {
  ...
  protected final TypeAliasRegistry typeAliasRegistry = new TypeAliasRegistry();
  ...
  public Configuration() {
    ...
  
    //日志類型別名
    typeAliasRegistry.registerAlias("SLF4J", Slf4jImpl.class);
    typeAliasRegistry.registerAlias("COMMONS_LOGGING", JakartaCommonsLoggingImpl.class);
    typeAliasRegistry.registerAlias("LOG4J", Log4jImpl.class);
    typeAliasRegistry.registerAlias("LOG4J2", Log4j2Impl.class);
    typeAliasRegistry.registerAlias("JDK_LOGGING", Jdk14LoggingImpl.class);
    typeAliasRegistry.registerAlias("STDOUT_LOGGING", StdOutImpl.class);
    typeAliasRegistry.registerAlias("NO_LOGGING", NoLoggingImpl.class);
  
    typeAliasRegistry.registerAlias("CGLIB", CglibProxyFactory.class);
    typeAliasRegistry.registerAlias("JAVASSIST", JavassistProxyFactory.class);
  
    languageRegistry.setDefaultDriverClass(XMLLanguageDriver.class);
    languageRegistry.register(RawLanguageDriver.class);
  }
}

從上面的Configuration構(gòu)造函數(shù)中可以看到,類型別名中定義了日志實(shí)現(xiàn)的名稱與,實(shí)現(xiàn)類.class(這些實(shí)現(xiàn)類都是MyBatis實(shí)現(xiàn)的,稍后再說(shuō))

//設(shè)置日志實(shí)現(xiàn)類
public void setLogImpl(Class<? extends Log> logImpl) {
  if (logImpl != null) {
    this.logImpl = logImpl;
    //調(diào)用日志工廠,設(shè)置日志實(shí)現(xiàn)
    LogFactory.useCustomLogging(this.logImpl);
  }
}

BaseBuilder

構(gòu)造函數(shù)

public BaseBuilder(Configuration configuration) {
  this.configuration = configuration;
  //可以看到這個(gè)地方的類型別名是從configuration中獲取的
  this.typeAliasRegistry = this.configuration.getTypeAliasRegistry();
  this.typeHandlerRegistry = this.configuration.getTypeHandlerRegistry();
}
protected <T> Class<? extends T> resolveClass(String alias) {
  if (alias == null) {
   return null;
  }
  try {
   //解析別名
   return resolveAlias(alias);
  } catch (Exception e) {
   throw new BuilderException("Error resolving class. Cause: " + e, e);
  }
}

//這個(gè)不就是從別名中獲取對(duì)應(yīng)的實(shí)現(xiàn)嗎??!
//配置SLF4J,返回Slf4jImpl.class
protected <T> Class<? extends T> resolveAlias(String alias) {
  //這個(gè)地方的typeAliasRegistry是從configuration中獲取的
  return typeAliasRegistry.resolveAlias(alias);
}

TypeAliasRegistry 類型別名

類型別名注冊(cè)

//就是將key轉(zhuǎn)換位小寫,存入HashMap中,key是別名,value是Class類對(duì)象
public void registerAlias(String alias, Class<?> value) {
  if (alias == null) {
    throw new TypeException("The parameter alias cannot be null");
  }
  // issue #748
  String key = alias.toLowerCase(Locale.ENGLISH);
  if (TYPE_ALIASES.containsKey(key) && TYPE_ALIASES.get(key) != null && !TYPE_ALIASES.get(key).equals(value)) {
    throw new TypeException("The alias '" + alias + "' is already mapped to the value '" + TYPE_ALIASES.get(key).getName() + "'.");
  }
  TYPE_ALIASES.put(key, value);
}
//根據(jù)對(duì)應(yīng)的別名key,獲取實(shí)現(xiàn)類class
public <T> Class<T> resolveAlias(String string) {
  try {
    if (string == null) {
      return null;
    }
    // issue #748
    String key = string.toLowerCase(Locale.ENGLISH);
    Class<T> value;
    if (TYPE_ALIASES.containsKey(key)) {
      value = (Class<T>) TYPE_ALIASES.get(key);
    } else {
      value = (Class<T>) Resources.classForName(string);
    }
    return value;
  } catch (ClassNotFoundException e) {
    throw new TypeException("Could not resolve type alias '" + string + "'. Cause: " + e, e);
  }
}

Slf4jImpl Slf4j實(shí)現(xiàn)類

public class Slf4jImpl implements Log {

 private Log log;

 public Slf4jImpl(String clazz) {
  //使用Slf4j的API獲取日志器
  Logger logger = LoggerFactory.getLogger(clazz);

  if (logger instanceof LocationAwareLogger) {
   try {
    // check for slf4j >= 1.6 method signature
    logger.getClass().getMethod("log", Marker.class, String.class, int.class, String.class, Object[].class, Throwable.class);
    log = new Slf4jLocationAwareLoggerImpl((LocationAwareLogger) logger);
    return;
   } catch (SecurityException e) {
    // fail-back to Slf4jLoggerImpl
   } catch (NoSuchMethodException e) {
    // fail-back to Slf4jLoggerImpl
   }
  }

  // Logger is not LocationAwareLogger or slf4j version < 1.6
  log = new Slf4jLoggerImpl(logger);
 }

 ...
}

可見,最終還是調(diào)用具體的日志API實(shí)現(xiàn)!

LogFactory 日志工廠

package org.apache.ibatis.logging;

import java.lang.reflect.Constructor;

/**
 * 日志工廠
 */
public final class LogFactory {

 /**
  * Marker to be used by logging implementations that support markers
  */
 public static final String MARKER = "MYBATIS";

 // 記錄當(dāng)前使用的第三方日志庫(kù)組件所對(duì)應(yīng)的適配器的方法
 private static Constructor<? extends Log> logConstructor;

 // tryImplementation 進(jìn)行嘗試加載
 static {
  tryImplementation(LogFactory::useSlf4jLogging);
  tryImplementation(LogFactory::useCommonsLogging);
  tryImplementation(LogFactory::useLog4J2Logging);
  tryImplementation(LogFactory::useLog4JLogging);
  tryImplementation(LogFactory::useJdkLogging);
  tryImplementation(LogFactory::useNoLogging);
 }

 // 私有化
 private LogFactory() {
  // disable construction
 }

 public static Log getLog(Class<?> aClass) {
  return getLog(aClass.getName());
 }

 public static Log getLog(String logger) {
  try {
   return logConstructor.newInstance(logger);
  } catch (Throwable t) {
   throw new LogException("Error creating logger for logger " + logger + ". Cause: " + t, t);
  }
 }

 /**
  * 以下的 useXXXogging 的方法都是嘗試加載日志的實(shí)現(xiàn)
  * 最終的實(shí)現(xiàn)都是 setImplementation
  */


 public static synchronized void useCustomLogging(Class<? extends Log> clazz) {
  setImplementation(clazz);
 }

 public static synchronized void useSlf4jLogging() {
  setImplementation(org.apache.ibatis.logging.slf4j.Slf4jImpl.class);
 }

 public static synchronized void useCommonsLogging() {
  setImplementation(org.apache.ibatis.logging.commons.JakartaCommonsLoggingImpl.class);
 }

 public static synchronized void useLog4JLogging() {
  setImplementation(org.apache.ibatis.logging.log4j.Log4jImpl.class);
 }

 public static synchronized void useLog4J2Logging() {
  setImplementation(org.apache.ibatis.logging.log4j2.Log4j2Impl.class);
 }

 public static synchronized void useJdkLogging() {
  setImplementation(org.apache.ibatis.logging.jdk14.Jdk14LoggingImpl.class);
 }

 public static synchronized void useStdOutLogging() {
  setImplementation(org.apache.ibatis.logging.stdout.StdOutImpl.class);
 }

 public static synchronized void useNoLogging() {
  setImplementation(org.apache.ibatis.logging.nologging.NoLoggingImpl.class);
 }

 /**
  * 嘗試加載
  * @param runnable
  */
 private static void tryImplementation(Runnable runnable) {
  if (logConstructor == null) {
   try {
    // 會(huì)調(diào)用 useSlf4jLogging 類似的方法
    runnable.run();
   } catch (Throwable t) {
    // ignore
   }
  }
 }

 /**
  * 設(shè)計(jì)日志的實(shí)現(xiàn)類
  * @param implClass Log 的子類
  */
 private static void setImplementation(Class<? extends Log> implClass) {
  try {
   // 通過(guò)反射獲取構(gòu)造方法
   Constructor<? extends Log> candidate = implClass.getConstructor(String.class);
   Log log = candidate.newInstance(LogFactory.class.getName());
   if (log.isDebugEnabled()) {
    log.debug("Logging initialized using '" + implClass + "' adapter.");
   }
   //設(shè)置日志實(shí)現(xiàn)的有參構(gòu)造函數(shù)
   logConstructor = candidate;
  } catch (Throwable t) {
   throw new LogException("Error setting Log implementation. Cause: " + t, t);
  }
 }

}

可以看到其中有靜態(tài)代碼塊,隨著類的加載,嘗試加載日志的實(shí)現(xiàn)!

總結(jié)

通過(guò)以上的一步一步分析,可以看到MyBatis是分別通過(guò)對(duì)日志實(shí)現(xiàn)進(jìn)行進(jìn)行包裝,最終還是調(diào)用具體的日志API實(shí)現(xiàn).

好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。

相關(guān)文章

  • 大數(shù)據(jù) java hive udf函數(shù)的示例代碼(手機(jī)號(hào)碼脫敏)

    大數(shù)據(jù) java hive udf函數(shù)的示例代碼(手機(jī)號(hào)碼脫敏)

    這篇文章主要介紹了大數(shù)據(jù) java hive udf函數(shù)(手機(jī)號(hào)碼脫敏),的相關(guān)知識(shí),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-06-06
  • Java?MyBatis實(shí)戰(zhàn)之QueryWrapper中and和or拼接技巧大全

    Java?MyBatis實(shí)戰(zhàn)之QueryWrapper中and和or拼接技巧大全

    在Java中QueryWrapper是MyBatis-Plus框架中的一個(gè)查詢構(gòu)造器,它提供了豐富的查詢方法,其中包括and和or方法,可以用于構(gòu)建復(fù)雜的查詢條件,這篇文章主要給大家介紹了關(guān)于Java?MyBatis實(shí)戰(zhàn)之QueryWrapper中and和or拼接技巧的相關(guān)資料,需要的朋友可以參考下
    2024-07-07
  • 創(chuàng)建SpringBoot工程并集成Mybatis的方法

    創(chuàng)建SpringBoot工程并集成Mybatis的方法

    這篇文章主要介紹了創(chuàng)建SpringBoot工程并集成Mybatis,需要的朋友可以參考下
    2018-06-06
  • Java中double和float類型的區(qū)別與使用方法

    Java中double和float類型的區(qū)別與使用方法

    float和double都是用來(lái)表示浮點(diǎn)數(shù)的數(shù)據(jù)類型,但是它們之間有一些區(qū)別,這篇文章主要給大家介紹了關(guān)于Java中double和float類型的區(qū)別與使用方法的相關(guān)資料,需要的朋友可以參考下
    2024-07-07
  • Java創(chuàng)建線程三種方式的優(yōu)缺點(diǎn)

    Java創(chuàng)建線程三種方式的優(yōu)缺點(diǎn)

    今天小編就為大家分享一篇關(guān)于Java創(chuàng)建線程三種方式的優(yōu)缺點(diǎn),小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2018-12-12
  • springboot如何為web層添加統(tǒng)一請(qǐng)求前綴

    springboot如何為web層添加統(tǒng)一請(qǐng)求前綴

    這篇文章主要介紹了springboot如何為web層添加統(tǒng)一請(qǐng)求前綴,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-02-02
  • 解決IDEA使用Spring Initializr創(chuàng)建項(xiàng)目時(shí)無(wú)法連接到https://start.spring.io的問(wèn)題

    解決IDEA使用Spring Initializr創(chuàng)建項(xiàng)目時(shí)無(wú)法連接到https://start.spring.io的問(wèn)

    這篇文章主要介紹了解決IDEA使用Spring Initializr創(chuàng)建項(xiàng)目時(shí)無(wú)法連接到https://start.spring.io的問(wèn)題,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-04-04
  • 一文搞懂設(shè)計(jì)模式中的單例模式

    一文搞懂設(shè)計(jì)模式中的單例模式

    這篇文章主要介紹了一文搞懂設(shè)計(jì)模式中的單例模式,單例模式是最簡(jiǎn)單的設(shè)計(jì)模式之一,屬于創(chuàng)建型模式,它提供了一種創(chuàng)建對(duì)象的方式,確保只有單個(gè)對(duì)象被創(chuàng)建,需要的朋友可以參考下
    2023-08-08
  • 詳解Java數(shù)據(jù)庫(kù)連接池

    詳解Java數(shù)據(jù)庫(kù)連接池

    今天繼續(xù)Java的課題,兩天沒(méi)有做任何事情,過(guò)了個(gè)自在的周末,但是不知道為什么總是有點(diǎn)淡淡的憂桑.之前游戲服務(wù)器的數(shù)據(jù)源使用的是阿里巴巴的Druid,今天就大概說(shuō)說(shuō)數(shù)據(jù)源,給個(gè)實(shí)例,需要的朋友可以參考下
    2021-06-06
  • 一文解決pom.xml報(bào)錯(cuò)Dependency "xxx" not found的問(wèn)題

    一文解決pom.xml報(bào)錯(cuò)Dependency "xxx" not f

    我們?cè)谑褂胢aven進(jìn)行jar包管理時(shí)有時(shí)會(huì)遇到pom.xml中報(bào)錯(cuò)Dependency “XXX” not found,所以在本文中將給大家介紹一下pom.xml報(bào)錯(cuò)Dependency "xxx" not found的解決方案,需要的朋友可以參考下
    2024-01-01

最新評(píng)論