浅谈日志框架slf4j原理,以及与logback,log4j的关系

# 前言

日志是每个Java项目必不可少的组成部分,我们几乎每天都和日志打交道。但是有的项目是logback,有的是log4j,有时候又是slf4j,傻傻分不清楚。

如果一个Spring项目原先是logback,合并一个新项目,新项目用的是log4j,那么日志文件用哪个,如果都用会怎么样?

下面就来说说。

slf4j,是个壳子,在java里面叫门面模式,顾名思义,就是一个代理的门面。它负责提供日志输出的标准方法,我们只需要调用slf4j的Logger和api,即可实现我们输出日志的功能。而至于具体日志输出的实现,则交给slf4j绑定的日志框架。log4j和logback都是更加底层一点的日志框架。其中logback是slf4j的默认实现,而log4j则要经过一层适配,才可以接入进来。

下面的图很清晰地表示了它们的关系:


一般使用日志如下:引用的是slf4j的类。

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
private static Loggerlog = LoggerFactory.getLogger(DemoApplication.class);

进入Logger可以看到,这是一个接口,并不负责具体实现,具体实现要看Logger里面的代码了。

# 具体实现

// LoggerFactory.getLogger第一步
Logger logger = getLogger(clazz.getName());
//进入后代码如下,return是返回工厂方法,进入getILoggerFactory
ILoggerFactory iLoggerFactory = getILoggerFactory();
return iLoggerFactory.getLogger(name);
// 判断初始化状态以及加锁,核心代码在performInitialization()方法中
if(INITIALIZATION_STATE == 0) {
    Class var0 = LoggerFactory.class;
    synchronized(LoggerFactory.class) {
        if(INITIALIZATION_STATE == 0) {
            INITIALIZATION_STATE = 1;
            performInitialization();
        }
    }
}
// performInitialization方法中,bind 方法负责绑定具体的实现类
bind();
if(INITIALIZATION_STATE == 3) {
    versionSanityCheck();
}
// 判断是否有多绑定
if(!isAndroid()) {
    e = findPossibleStaticLoggerBinderPathSet();
    reportMultipleBindingAmbiguity(e);
}     
/* 在findPossibleStaticLoggerBinderPathSet中,slf4j会扫描配置类org/slf4j/impl/StaticLoggerBinder.class,
如果classpath下扫描到多个配置类,说明有多个日志框架可以绑定,slf4j会选择其中一个作为最终的日志实现。
在某些时候我们会在启动日志中发现红色的slf4j的日志,就是类似这个地方通过System.err输出出来的。*/
try {
    ClassLoader ioe = LoggerFactory.class.getClassLoader();
    Enumeration paths;
    if(ioe == null) {
        paths = ClassLoader.getSystemResources(STATIC_LOGGER_BINDER_PATH);
    } else {
        paths = ioe.getResources(STATIC_LOGGER_BINDER_PATH);
    }
    while(paths.hasMoreElements()) {
        URL path = (URL)paths.nextElement();
        staticLoggerBinderPathSet.add(path);
    }
} catch (IOException var4) {
    Util.report("Error getting resources from path", var4);
}
//核心代码在StaticLoggerBinder中,这个类通过静态方法init初始化,内部持有一个自身对象,通过静态方法获取单例
StaticLoggerBinder.getSingleton();
INITIALIZATION_STATE = 3;
reportActualBinding(e);

在StaticLoggerBinder的init方法中,通过autoConfig加载日志配置文件。

先去找系统环境变量配置的logback.configurationFile,如果没有,则去找logback-test.xml,如果还是没有找到,就去找logback.xml。

最后依然没有找到,则初始化一个控制台ConsoleAppender,就像spring-boot项目初始化没有新增logback配置文件时的表现一样。

那么回到最开始的问题:

如果项目里同时绑定log4j和logback,slf4j会怎么样呢?他会选择其中一个作为实现方式,多余的则没有生效。

例如我们在spring的pom里面配置log4j的依赖,启动后,控制台会输出如下日志:

```

SLF4J: Class path contains multiple SLF4J bindings.

SLF4J: Found binding in [jar:file:/F:/mvn/repository/ch/qos/logback/logback-classic/1.2.11/logback-classic-1.2.11.jar!/org/slf4j/impl/StaticLoggerBinder.class]

SLF4J: Found binding in [jar:file:/F:/mvn/repository/org/slf4j/slf4j-log4j12/1.7.30/slf4j-log4j12-1.7.30.jar!/org/slf4j/impl/StaticLoggerBinder.class]

SLF4J: See http://www.slf4j.org/codes.html#multiple_bindings for an explanation.

SLF4J: Actual binding is of type [ch.qos.logback.classic.util.ContextSelectorStaticBinder]

```

如果觉得写得不错,不妨点个赞,您的肯定是我继续分享的动力,后续还会继续研究其他问题。

原文链接:https://blog.csdn.net/qq_29558011/article/details/125882115

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容