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

淺談slf4j中的橋接器是如何運作的

 更新時間:2020年12月03日 12:17:31   作者:程序新視界  
這篇文章主要介紹了slf4j中的橋接器是如何運作的,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

閱讀分析 slf4j 的日志源碼,發(fā)現(xiàn)其中涵蓋了許多知識點和優(yōu)秀的設(shè)計,關(guān)鍵它們都是活生生的實踐案例。寫成系列文章與大家分享,歡迎持續(xù)關(guān)注。

前言

在日志框架 slf4j 中有一組項目,除了核心的 slf4j-api 之外,還有 slf4j-log4j12、slf4j-jdk14 等項目。這一類項目統(tǒng)稱橋接器項目,針對不同的日志框架有不同的橋接器項目。

在使用 logback 日志框架時,并沒有針對的橋接器,這是因為 logback 與 slf4j 是一個作者所寫,在 logback 中直接實現(xiàn)了 slf4j 的 SPI 機制。

但如果使用其他日志框架,那么就必須要用到橋機器相關(guān)依賴。比如,當(dāng)我們基于 log4j 使用 slf4j 時,除了需要引入 log4j 的 jar 包依賴,還需要引入 slf4j 的下面兩個依賴:

<dependency>
 <groupId>org.slf4j</groupId>
 <artifactId>slf4j-api</artifactId>
</dependency>
<dependency>
 <groupId>org.slf4j</groupId>
 <artifactId>slf4j-log4j12</artifactId>
</dependency>

slf4j-api 為核心依賴,必須引入,而 slf4j-log4j12 就是橋接器用來在 slf4j 和 log4j 之間進(jìn)行過渡和封裝。下面,我們就聊聊橋接器項目的核心實現(xiàn)。

slf4j-log4j12 橋接器的價值

要了解橋接器的運作,首先需要回顧一下 slf4j 的 SPI 機制。在我們通過 LoggerFactory.getLogger(Foo.class); 時,slf4j 會通過 SPI 機制尋找并初始化 SLF4JServiceProvider 的實現(xiàn)類。

然后,通過 SLF4JServiceProvider 的實現(xiàn)類來獲取日志相關(guān)的具體工廠類對象,進(jìn)而進(jìn)行日志功能的處理。先來看一下 SLF4JServiceProvider 的接口定義:

public interface SLF4JServiceProvider {

  /**
   * 返回ILoggerFactory的實現(xiàn)類,用于LoggerFactory類的綁定
   */
  ILoggerFactory getLoggerFactory();

  /**
   * 返回IMarkerFactory實例
   */
  IMarkerFactory getMarkerFactory();

  /**
   * 返回MDCAdapter實例
   */
  MDCAdapter getMDCAdapter();

  /**
   * 獲取請求版本
   */
  String getRequesteApiVersion();

  /**
   * 初始化,實現(xiàn)類中一般用于初始化ILoggerFactory等
   */
  void initialize();
}

SLF4JServiceProvider 接口是在 slf4j-api 中定義的,具體的實現(xiàn)類由其他日志框架來完成。但是像 log4j(logback“敵對陣營”)是不會在框架內(nèi)實現(xiàn)該接口的。那么,怎么辦?

針對此問題,slf4j 提供了 slf4j-log4j12 這類橋接器的過渡項目。在其中實現(xiàn) SLF4JServiceProvider 接口,并對 Log4j 日志框架接口進(jìn)行封裝,將 Logger(slf4j) 接收到的命令全部委托給 Logger(log4j) 去完成,在使用者無感知的情況下完成偷天換日。

slf4j-log4j12 的核心實現(xiàn)類

理解了橋接器的存在價值及原理,下面就來看看 slf4j-log4j12 是如何實現(xiàn)這一功能的。

首先來看看核心實現(xiàn)類之一 Log4j12ServiceProvider。它實現(xiàn)了 SLF4JServiceProvider 接口,主要功能就是完成接口中定義的相關(guān)工廠接口的實現(xiàn)。源代碼如下:

public class Log4j12ServiceProvider implements SLF4JServiceProvider {

  public static String REQUESTED_API_VERSION = "1.8.99"; 

  private ILoggerFactory loggerFactory; 
  private IMarkerFactory markerFactory; 
  private MDCAdapter mdcAdapter;
  
  public Log4j12ServiceProvider() {
    try {
      @SuppressWarnings("unused")
      Level level = Level.TRACE;
    } catch (NoSuchFieldError nsfe) {
      Util.report("This version of SLF4J requires log4j version 1.2.12 or later. See also http://www.slf4j.org/codes.html#log4j_version");
    }
  }

  @Override
  public void initialize() {
    loggerFactory = new Log4jLoggerFactory();
    markerFactory = new BasicMarkerFactory();
    mdcAdapter = new Log4jMDCAdapter();
  }
  
  @Override
  public ILoggerFactory getLoggerFactory() {
    return loggerFactory;
  }

  @Override
  public IMarkerFactory getMarkerFactory() {
    return markerFactory;
  }

  @Override
  public MDCAdapter getMDCAdapter() {
    return mdcAdapter;
  }

  @Override
  public String getRequesteApiVersion() {
    return REQUESTED_API_VERSION;
  }
}



該類的實現(xiàn)看起來很簡單,構(gòu)造方法中通過嘗試使用 log4j 的 Level.TRACE 調(diào)用來驗證 log4j 的版本是否符合要求。log4j1.2.12 之前并沒有 Level.TRACE,所以會拋出異常,并打印日志信息。不得不贊嘆作者在此處檢查版本的巧妙用法。

而這里對接口中返回的實現(xiàn)類主要通過 initialize() 方法來實現(xiàn)的。這里我們重點看 Log4jLoggerFactory 類的實現(xiàn)。

public class Log4jLoggerFactory implements ILoggerFactory {

  private static final String LOG4J_DELEGATION_LOOP_URL = "http://www.slf4j.org/codes.html#log4jDelegationLoop";

  // check for delegation loops
  static {
    try {
      Class.forName("org.apache.log4j.Log4jLoggerFactory");
      String part1 = "Detected both log4j-over-slf4j.jar AND bound slf4j-log4j12.jar on the class path, preempting StackOverflowError. ";
      String part2 = "See also " + LOG4J_DELEGATION_LOOP_URL + " for more details.";

      Util.report(part1);
      Util.report(part2);
      throw new IllegalStateException(part1 + part2);
    } catch (ClassNotFoundException e) {
      // this is the good case
    }
  }

  ConcurrentMap<String, Logger> loggerMap;

  public Log4jLoggerFactory() {
    loggerMap = new ConcurrentHashMap<>();
    // force log4j to initialize
    org.apache.log4j.LogManager.getRootLogger();
  }

  @Override
  public Logger getLogger(String name) {
    Logger slf4jLogger = loggerMap.get(name);
    if(slf4jLogger != null) {
      return slf4jLogger;
    } else {
      org.apache.log4j.Logger log4jLogger;
      if(name.equalsIgnoreCase(Logger.ROOT_LOGGER_NAME)) {
        log4jLogger = LogManager.getRootLogger();
      } else {
        log4jLogger = LogManager.getLogger(name);
      }

      Logger newInstance = new Log4jLoggerAdapter(log4jLogger);
      Logger oldInstance = loggerMap.putIfAbsent(name, newInstance);
      return oldInstance == null ? newInstance : oldInstance;
    }
  }
}

在 Log4j12ServiceProvider 中進(jìn)行了 Log4jLoggerFactory 的實例化操作,也就直接 new 出來一個對象。我們知道,在 new 對象執(zhí)行會先執(zhí)行 static 代碼塊,本類的靜態(tài)代碼塊的核心工作就是檢查依賴文件中是否同時存在反向橋接器的依賴。

其中,org.apache.log4j.Log4jLoggerFactory 是反向橋接器 log4j-over-slf4j 項目中的類,如果加裝到了,說明存在,則拋出異常,打印日志信息。此處再次贊嘆作者運用的技巧的巧妙。

在 Log4jLoggerFactory 的構(gòu)造方法中,做了兩件事:第一,初始化一個 ConcurrentMap 變量,用于存儲實例化的 Logger;第二,強制初始化 log4j 的組件,其中強制初始化 log4j 的組件是通過 getRootLogger 方法,來初始化一些靜態(tài)的變量。

構(gòu)造方法時初始化了 ConcurrentMap 變量,在 Log4jLoggerFactory 實現(xiàn)的 getLogger 方法中,先從 Map 中獲取一下是否存在對應(yīng)的 Logger,如果存在直接返回,如果不存在則進(jìn)行構(gòu)造。而構(gòu)造的 Log4jLoggerAdapter 類很顯然使用了適配器模式,它內(nèi)部持有了 log4j 的 Logger 對象,自身又實現(xiàn)了 slf4j 的 Logger 接口。

下面看一下 Log4jLoggerAdapter 的部分代碼實現(xiàn):

public final class Log4jLoggerAdapter extends LegacyAbstractLogger implements LocationAwareLogger, Serializable {

  final transient org.apache.log4j.Logger logger;

  Log4jLoggerAdapter(org.apache.log4j.Logger logger) {
    this.logger = logger;
    this.name = logger.getName();
    traceCapable = isTraceCapable();
  }

  @Override
  public boolean isDebugEnabled() {
    return logger.isDebugEnabled();
  }


  @Override
  public void log(Marker marker, String callerFQCN, int level, String msg, Object[] arguments, Throwable t) {
    Level log4jLevel = toLog4jLevel(level);
    NormalizedParameters np = NormalizedParameters.normalize(msg, arguments, t);
    String formattedMessage = MessageFormatter.basicArrayFormat(np.getMessage(), np.getArguments());
    logger.log(callerFQCN, log4jLevel, formattedMessage, np.getThrowable());
  }

  public void log(LoggingEvent event) {
    Level log4jLevel = toLog4jLevel(event.getLevel().toInt());
    if (!logger.isEnabledFor(log4jLevel))
      return;

    org.apache.log4j.spi.LoggingEvent log4jevent = toLog4jEvent(event, log4jLevel);
    logger.callAppenders(log4jevent);

  }
  
  // 省略其他方法
}

源碼中,通過構(gòu)造方法傳入 log4j 的 Logger 對象,而 Log4jLoggerAdapter 對外提供的方法,都是通過 log4j 的 Logger 進(jìn)行具體實現(xiàn)。

總之,slf4j 的 Logger 接口的方法通過 Log4jLoggerAdapter 進(jìn)行包裝和轉(zhuǎn)換,交由 log4j 的 Logger 去執(zhí)行,這就達(dá)到了連接 slf4j-api 和 log4j 的目的。而此時,slf4j-api 不并關(guān)系日志是如何實現(xiàn)記錄,對此也無感知。

小結(jié)

本文通過源碼跟蹤,逐步分析了 slf4j 項目中橋接器項目的運作機制,其中還涉及到了 SPI 機制、版本及依賴檢查小技巧、橋接器運作本質(zhì)(適配器模式)等。其實,在 slf4j 項目中還有文中提到的反向橋接器,其實基本機制也是如此,感興趣的朋友可以閱讀一下 log4j-over-slf4j 中的源碼。

到此這篇關(guān)于淺談slf4j中的橋接器是如何運作的 的文章就介紹到這了,更多相關(guān)slf4j 橋接器內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • java對接支付寶支付接口簡單步驟記錄

    java對接支付寶支付接口簡單步驟記錄

    最近項目APP需要接入微信、支付寶支付功能,在分配開發(fā)任務(wù)時,聽說微信支付接口比支付寶支付接口要難實現(xiàn),這篇文章主要給大家介紹了關(guān)于java對接支付寶支付接口的簡單步驟,需要的朋友可以參考下
    2024-05-05
  • 使用Logback日志保存到相對路徑的操作

    使用Logback日志保存到相對路徑的操作

    這篇文章主要介紹了使用Logback日志保存到相對路徑的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-11-11
  • Java如何正確的使用wait-notify方法你知道嗎

    Java如何正確的使用wait-notify方法你知道嗎

    這篇文章主要為大家詳細(xì)介紹了Java如何正確的使用wait-notify方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-03-03
  • 詳解SpringBoot是如何保證接口安全的

    詳解SpringBoot是如何保證接口安全的

    對于互聯(lián)網(wǎng)來說,只要你系統(tǒng)的接口會暴露在外網(wǎng),就避免不了接口安全問題。?如果你的接口在外網(wǎng)裸奔,只要讓黑客知道接口的地址和參數(shù)就可以調(diào)用,那簡直就是災(zāi)難。這篇文章主要介紹了SpringBoot保證接口安全的方法,需要的可以參考一下
    2023-02-02
  • Spring中的@Scheduled源碼解析

    Spring中的@Scheduled源碼解析

    這篇文章主要介紹了Spring中的@Scheduled源碼解析,定時任務(wù)調(diào)度的基礎(chǔ)是ScheduledAnnotationBeanPostProcessor類,這是一個實現(xiàn)了BeanPostProcessor接口的后置處理器,需要的朋友可以參考下
    2023-09-09
  • Java中的異步與線程池解讀

    Java中的異步與線程池解讀

    這篇文章主要介紹了Java中的異步與線程池,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-11-11
  • 利用反射實現(xiàn)Excel和CSV 轉(zhuǎn)換為Java對象功能

    利用反射實現(xiàn)Excel和CSV 轉(zhuǎn)換為Java對象功能

    將Excel或CSV文件轉(zhuǎn)換為Java對象(POJO)以及將Java對象轉(zhuǎn)換為Excel或CSV文件可能是一個復(fù)雜的過程,但如果使用正確的工具和技術(shù),這個過程就會變得十分簡單,在本文中,我們將了解如何利用一個Java反射的庫來實現(xiàn)這個功能,需要的朋友可以參考下
    2023-11-11
  • idea安裝hsdis的方法

    idea安裝hsdis的方法

    這篇文章主要介紹了idea安裝hsdis,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-03-03
  • Java對象布局(JOL)實現(xiàn)過程解析

    Java對象布局(JOL)實現(xiàn)過程解析

    這篇文章主要介紹了Java對象布局(JOL)實現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-04-04
  • Intellij Idea修改代碼方法參數(shù)自動提示快捷鍵的操作

    Intellij Idea修改代碼方法參數(shù)自動提示快捷鍵的操作

    這篇文章主要介紹了Intellij Idea修改代碼方法參數(shù)自動提示快捷鍵的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-01-01

最新評論