说说Java日志那些事

news2024/12/23 12:46:36

日志是每个应用程序在开发的过程中必不可少的组件,通过日志输出可以获取项目的一些运行信息,监控项目的运行指标等,同时,通过学习Java日志框架还可以学习到桥接器与适配器模式等。

日志框架

  • log4j (日志实现)
    log4j是最早出现的一个日志框架,由Apache组织开源,相应的依赖为:
        <dependency>
            <groupId>log4j</groupId>
            <artifactId>log4j</artifactId>
            <version>1.2.17</version>
        </dependency>
  • jul (日志实现)
    jul在log4j之后才出现,是JDK自带的一个日志包,主要是怕影响了日志市场。jul的使用与log4j大同小异,主要的差异在于如果使用jul,则不需要导入额外的jar包。

  • jcl (jakatar commons logging 日志门面)
    jcl是一个日志门面,也就是日志的抽象层与接口层,jcl不提供具体的日志实现,而是提供接口层,jcl中内置了许多日志实现类,在获取Logger对象时,jcl会依次加载一些实现类,如果有classpath下有实现,且没有抛出NoClassDefFoundError,则就会使用当前加载到的日志实现。

在这里插入图片描述

  • logback(日志实现)
    logack同log4j一样,出自同一个人之手,同样都是日志实现,但性能超过log4j,作者在设计logback的时候想到了log4j,于是也开发了一个日志抽象层,也就是日志门面。logack的依赖如下
<!-- https://mvnrepository.com/artifact/ch.qos.logback/logback-core -->
<dependency>
    <groupId>ch.qos.logback</groupId>
    <artifactId>logback-core</artifactId>
    <version>1.4.7</version>
</dependency>

logback-core是logback的核心包,实现了logback的所有功能。还有一个包为logback-classic,这个包除了引入了logback的核心包,还引入了适配slf4j的一些包。

<!-- https://mvnrepository.com/artifact/ch.qos.logback/logback-classic -->
<dependency>
    <groupId>ch.qos.logback</groupId>
    <artifactId>logback-classic</artifactId>
    <version>1.4.6</version>
</dependency>
  • slf4j(日志门面)
    slf4j与logback一样都是一个人写的,兼容起来比较合适。slf4j只提供日志门面,不负责具体的日志实现,具体的日志实现原理为StaticLoggerBinder这个类,每一个实现了slf4j门面日志实现的jar包中都应该有这个类,只要一加载这个类,就会触发这个类的初始化和静态代码块方法,返回一个ILoggerFactory,这个对象负责具体的日志实现框架的创建。

slf4j日志门面的依赖如下:

        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-api</artifactId>
            <version>1.7.36</version>
        </dependency>
  • log4j2
    apache在看到logback的 成功之后,借鉴了logback的实现思想,写了一个全新的日志框架,名为log4j2。log4j2的性能优于logback,除此之外,log4j2还是一个日志门面,由此可见,log4j2真是狼子野心,不仅提供了日志实现,还想做全新的日志门面,但是虽然它提供了日志门面,市面上绝大多数项目还是用slf4j做日志门面,至于日志实现,就看项目中选择用哪个jar包了。

log4j2的日志门面

<!-- https://mvnrepository.com/artifact/org.apache.logging.log4j/log4j-api -->
<dependency>
    <groupId>org.apache.logging.log4j</groupId>
    <artifactId>log4j-api</artifactId>
    <version>2.19.0</version>
</dependency>

log4j2的日志实现

<!-- https://mvnrepository.com/artifact/org.apache.logging.log4j/log4j-core -->
<dependency>
    <groupId>org.apache.logging.log4j</groupId>
    <artifactId>log4j-core</artifactId>
    <version>2.18.0</version>
</dependency>

日志适配包

我们知道,slf4j是个日志门面,要想在项目中进行具体的日志输出,肯定还是要靠日志实现的,由于 slf4j与logback的作者都是同一个人,只要引入了logback那么就会自动实现到logback。那其他的日志框架是怎么桥接到slf4j呢?slf4j的核心代码在这里

    public static Logger getLogger(String name) {
        ILoggerFactory iLoggerFactory = getILoggerFactory();
        return iLoggerFactory.getLogger(name);
    }
    private static final void bind() {
        try {
            String msg;
            try {
                Set<URL> staticLoggerBinderPathSet = null;
                if (!isAndroid()) {
                    staticLoggerBinderPathSet = findPossibleStaticLoggerBinderPathSet();
                    reportMultipleBindingAmbiguity(staticLoggerBinderPathSet);
                }

                StaticLoggerBinder.getSingleton();
                INITIALIZATION_STATE = 3;
                reportActualBinding(staticLoggerBinderPathSet);
            } catch (NoClassDefFoundError var7) {
                msg = var7.getMessage();
                if (!messageContainsOrgSlf4jImplStaticLoggerBinder(msg)) {
                    failedBinding(var7);
                    throw var7;
                }

                INITIALIZATION_STATE = 4;
                Util.report("Failed to load class \"org.slf4j.impl.StaticLoggerBinder\".");
                Util.report("Defaulting to no-operation (NOP) logger implementation");
                Util.report("See http://www.slf4j.org/codes.html#StaticLoggerBinder for further details.");
            } catch (NoSuchMethodError var8) {
                msg = var8.getMessage();
                if (msg != null && msg.contains("org.slf4j.impl.StaticLoggerBinder.getSingleton()")) {
                    INITIALIZATION_STATE = 2;
                    Util.report("slf4j-api 1.6.x (or later) is incompatible with this binding.");
                    Util.report("Your binding is version 1.5.5 or earlier.");
                    Util.report("Upgrade your binding to version 1.6.x.");
                }

                throw var8;
            } catch (Exception var9) {
                failedBinding(var9);
                throw new IllegalStateException("Unexpected initialization failure", var9);
            }
        } finally {
            postBindCleanUp();
        }

    }

slf4j是直接调用了classpath下的StaticLoggerBinder.getSingleton();这个类的这个方法的。StaticLoggerBinder这个类在哪个包里呢?其实不是在slf4j-api这个包里,而是在每个与slf4j绑定的适配包中,也就是说,你导入了哪个日志框架与slf4j的适配包,这个适配包里就会有StaticLoggerBinder这个类,但是ILoggerFactory是每个日志框架提供的不同的实现,这样每个Factory工厂就可以跟据导入的实现生产不同的日志配置。
在这里插入图片描述
也就是说slf4j的适配包提供了日志工厂、slf4j的Logger的实现等。

  • jul - slf4j的适配包
<dependency>
		<groupId>org.slf4j</groupId>
		<artifactId>slf4j-jdk14</artifactId>
		<version>1.7.32</version>
	</dependency>

这是jul适配的相关类

在这里插入图片描述

  • logj4 - slf4j的适配包
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-log4j12</artifactId>
            <version>1.7.2</version>
        </dependency>

可以看到这个jar包的包名和类名与slf4j-jdk14是差不多的。
在这里插入图片描述

  • logback - slf4j
        <dependency>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-classic</artifactId>
            <version>1.2.8</version>
        </dependency>

logback的日志实现也是绑定了相关slf4j的类的。
在这里插入图片描述

  • log4j2 - slf4j
        <dependency>
            <groupId>org.apache.logging.log4j</groupId>
            <artifactId>log4j-slf4j-impl</artifactId>
            <version>2.12.4</version>
        </dependency>

log4j2适配slf4j

在这里插入图片描述

日志桥接包

什么是日志桥接包?因为slf4j出现的比较晚,很多的项目可能项目中直接用的是log4j、jul、jcl等,那么如果是用到了这些实现,该如何全部统一到slf4j + logback呢?
在这里插入图片描述

从这张官方图就可以看出来了,如果想要替换到项目中已有的jar包,则需要做以下三个步骤:

  1. 排除掉原有jar包 (例如排除掉其他框架引入的log4j)。
  2. 如果直接把原有jar包排除了,程序肯定就直接运行不起来了,所以此时需要导入这个jar包的桥接包,这个桥接包就是一个狸猫换太子包,偷天换日包,这个包模仿了原有实现包所有的包名和类名,但是确实掉的slf4j的API。
  3. 引入slf4j和具体的日志实现。
  • log4j 桥接到 slf4j
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>log4j-over-slf4j</artifactId>
            <version>1.7.32</version>
        </dependency>

模仿了log4j的包与类,伪造了log4j但是真正调用的slf4j。

  • jcl桥接到slf4j

有些项目可能没有用具体的日志实现,而是用的jcl作为日志门面,这个时候还是同样的道理,排除掉真正的jcl包,导入到jcl与slf4j的桥接包。

        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>jcl-over-slf4j</artifactId>
            <version>1.7.32</version>
        </dependency>

在这里插入图片描述
可以看到,虽然jar包名是叫jcl-over-slf4j,但是包名都是apache commons logging的,这个包也是和commons-logging这个包一模一样的包名和类名,但是调用的是Slf4j的API,包括原有的Log对象,也是适配了Slf4j的log对象。

在这里插入图片描述

  • jul桥接到slf4j

Java Logging API相关类由JDK提供,我们不能排除掉JDK,因而,在"Java Logging API转SLF4J"过程中采用的转换方案跟"JCL转SLF4J"和"Log4J转SLF4J"采用的转换方案不同,具体思路是以jul-to-slf4j提供的"org.slf4j.bridge.SLF4JBridgeHandler"替换掉Java Logging API使用的Handler(Handler即是Appender),Java Logging API原来使用的Handler将日志输出到Console,文件,数据库等,现在"org.slf4j.bridge.SLF4JBridgeHandler"将日志输出到SLF4J日志框架。简单来说,就是将原本Java Logging API的日志输出重定向到SLF4J日志框架。
我们看一下jul-over-slf4j的jar包实现

        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>jul-to-slf4j</artifactId>
            <version>1.7.32</version>
        </dependency>

Slf4jBridgeHandler

	//JUL发布日志的方法
    public void publish(LogRecord record) {
        if (record != null) {
            org.slf4j.Logger slf4jLogger = this.getSLF4JLogger(record);
            String message = record.getMessage();
            if (message == null) {
                message = "";
            }

            if (slf4jLogger instanceof LocationAwareLogger) {
                this.callLocationAwareLogger((LocationAwareLogger)slf4jLogger, record);
            } else {
                this.callPlainSLF4JLogger(slf4jLogger, record);
            }

        }
    }

//获取slf4j的logger
    protected org.slf4j.Logger getSLF4JLogger(LogRecord record) {
        String name = record.getLoggerName();
        if (name == null) {
            name = "unknown.jul.logger";
        }

        return LoggerFactory.getLogger(name);
    }


//真正调用slf4j的方法发布
    protected void callPlainSLF4JLogger(org.slf4j.Logger slf4jLogger, LogRecord record) {
        String i18nMessage = this.getMessageI18N(record);
        int julLevelValue = record.getLevel().intValue();
        if (julLevelValue <= TRACE_LEVEL_THRESHOLD) {
            slf4jLogger.trace(i18nMessage, record.getThrown());
        } else if (julLevelValue <= DEBUG_LEVEL_THRESHOLD) {
            slf4jLogger.debug(i18nMessage, record.getThrown());
        } else if (julLevelValue <= INFO_LEVEL_THRESHOLD) {
            slf4jLogger.info(i18nMessage, record.getThrown());
        } else if (julLevelValue <= WARN_LEVEL_THRESHOLD) {
            slf4jLogger.warn(i18nMessage, record.getThrown());
        } else {
            slf4jLogger.error(i18nMessage, record.getThrown());
        }
    }

jul-to-slf4j这个包相对来说比较特殊,因为jdk自带的包是没法排除的,那么这个时候slf4j设计了一个桥接器,这个桥接器是如何把jul的日志全部用slf4j输出的。SLF4JBridgeHandler 这个类设计了两个方法:

 public static void install() {
        LogManager.getLogManager().getLogger("").addHandler(new SLF4JBridgeHandler());
    }

    public static void removeHandlersForRootLogger() {
        Logger rootLogger = getRootLogger();
        Handler[] handlers = rootLogger.getHandlers();

        for(int i = 0; i < handlers.length; ++i) {
            rootLogger.removeHandler(handlers[i]);
        }

    }

removeHandlersForRootLogger 这个方法的作用是将rootLog的所有Handler全都移除,install方法的作用是将slf4j的桥接Handler添加到rootLog的Handler中。那这样的话,如果项目中有其他地方获取到Logger的话,只要不针对这个log做定制化设置,都是继承的rootLog,而rootLog在发布日志的时候,就是调用了Slf4j的API。

在这里插入图片描述
这行代码就是根据log的name获取日志对象,就可以具体桥接到某个日志实现框架了。

在这里插入图片描述
可以看到,这样就桥接到slf4j了,记得一定要先调用绑定器的两个方法。

而jul-to-slf4j 唯一无法解决的就是,如果针对jul的某个logger做了定制化设计,此时就桥接就失效了,因为jdk的源码是修改不了的。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/454620.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

上线控制台,降低使用门槛|Higress 1.0.0 RC 版本发布

作者&#xff1a;澄潭 前言 历时 5 个多月&#xff0c;Higress 推出了首个 RC &#xff08;Release Candidate&#xff0c;即正式发布的候选&#xff09;版本 1.0.0-rc&#xff0c;用户可以通过控制台&#xff0c;开箱即用地使用云原生网关。 选用 Higress 作为云原生网关的…

探究GPU同时做渲染与通用计算的并行性

在10年前&#xff0c;随着CUDA与OpenCL的纷纷出炉&#xff0c;GPGPU也着实热了一把。而现今&#xff0c;不少公司更是将GPGPU作为挖矿、搞机器学习的计算利器。于是乎&#xff0c;有许多言论声称GPU将很快取代CPU&#xff01;那么现代化的GPGPU是否具有如此强大的威力甚至于能取…

sql注入知识---时间盲注

MySQL手注之时间盲注详解 时间盲注原理&#xff1a;注意&#xff1a;基于时间盲注sleep函数函数用法&#xff1a;例子&#xff1a;观察以下语句 基于时间盲注if函数函数用法&#xff1a; 基本步骤&#xff1a;基于时间的盲注案例&#xff08;sqli-lab less-9&#xff09;枚举当…

【Vue】学习笔记-组件化编程

学习笔记-组件化编程 模块与组件、模块化与组件化非单文件组件基本使用组件注意事项组件的嵌套VueComponent一个重要的内置关系 单文件组件 模块与组件、模块化与组件化 模块 a.理解&#xff1a;向外提供特定功能的js程序&#xff0c;一般就是一个js文件 b.为什么&#xff1a;j…

实验8---SpringMVC基础

实验八 SpringMVC基础 一、实验目的及任务 通过该实验&#xff0c;掌握SpringMVC框架搭建步骤&#xff0c;掌握SpringMVC控制器的编写方法以及与spring依赖注入结合方法。 二、实验环境及条件 主机操作系统为Win10&#xff0c;Tomcat,j2sdk1.6或以上版本。 三、实验实施…

使用Linux运维常识

一.基础操作 1.终端常用快捷键 快捷键描述ctrl键盘左键向左跳一个单词ctrl键盘右键向右跳一个单词Ctrl c停止当前正在运行的命令。Ctrl z将当前正在运行的命令放入后台并暂停它的进程。Ctrl d关闭当前终端会话。Ctrl l清屏&#xff0c;也可以用clear命令实现Tab自动补全当…

倾斜摄影超大场景的三维模型转换3DTILES格式有哪些好处?

倾斜摄影超大场景的三维模型转换3DTILES格式有哪些好处&#xff1f; 倾斜摄影超大场景的三维模型转换到3D Tiles格式有以下几个好处&#xff1a; 1、数据存储效率高&#xff1a;3D Tiles是一种高效的地理数据存储格式&#xff0c;能够将大规模的倾斜摄影三维模型数据分块存储…

Hive312的计算引擎由MapReduce(默认)改为Spark(亲测有效)

一、Hive引擎包括&#xff1a;默认MR、tez、spark 在低版本的hive中&#xff0c;只有两种计算引擎mr, tez 在高版本的hive中&#xff0c;有三种计算引擎mr, spark, tez 二、Hive on Spark和Spark on Hive的区别 Hive on Spark&#xff1a;Hive既存储元数据又负责SQL的解析&…

【Linux】生产者消费者模型——环形队列RingQueue(信号量)

文章目录 铺垫信号量信号量概念信号量PV操作信号量基本接口 环形队列的生产消费模型引入环形队列访问环形队列代码实现代码改造多生产者多消费者代码 总结 铺垫 之前写的代码是存在不足的地方的&#xff1a; 我们使用线程操作临界资源的时候要先去判断临界资源是否满足条件&am…

QoS部署

1.总部和分部之间视频出现花屏,语音图像不同步的现象是有哪些原因导致的? 如图所示总部和分部之间的流量有FTP数据流量、视频流量和语音流量。如果在总部和分部之间的这几类流量没有做QoS部分或者优先级的区分,那么这些流量基于默认的无差别的流量策略去竞争带宽,如果FTP数…

暴躁兔分享:火爆圈子的XEN,我们如何吃到一波红利的

这周沉闷的熊市突然冲出一个XEN项目。 在这个项目上我们经历了拿到消息&#xff0c;看不懂不做&#xff0c;获取新的信息&#xff0c;发现可以搞&#xff0c;冲的这么一个过程。虽然由于信息查和认知差没有吃到最早拿到信息那波的利润&#xff0c;但是通过分析也跟着喝了一点汤…

网页源代码检查时隐藏 WebSocket 的后端地址

背景 近期在自研如何通过 OpenAI 实现 与ChatGPT 官网一样的聊天对话效果。 用到了 html5websocketpython 三项技术 , 于是用一天时间自学了一下这方面技术。 当实现了功能之后&#xff0c;就得考虑安全问题&#xff1a; 在用 html5 实现与 websocket 通讯时&#xff0c;如何保…

管理后台项目-05-SKU列表-上下架-详情抽屉效果-深度选择器

目录 1-SKU列表 2-SKU上下架 3-SKU详情 1-SKU列表 当用户点击Sku管理&#xff0c;组件挂载的时候&#xff0c;我们需要获取sku列表信息&#xff1b;但是获取列表方法在分页列表改变页码和每页显示大小的时候也需要触发&#xff0c;我们封装为一个方法。 //sku列表的接口 /adm…

Elasticsearch聚合、自动补全 | 黑马旅游

一、数据聚合 1、聚合的分类 聚合&#xff08;aggregations&#xff09;可以实现对文档数据的统计、分析、运算。 聚合常见有三类&#xff1a; 桶聚合 Bucket&#xff1a;对文档数据分组&#xff0c;并统计每组数量 TermAggregation&#xff1a;按照文档字段值分组&#xf…

中国电子学会2023年03月青少年软件编程Scratch图形化等级考试试卷二级真题(含答案)

中国电子学会2023年03月青少年软件编程Scratch图形化等级考试试卷二级 1.小猫的程序如图所示&#xff0c;积木块的颜色与球的颜色一致。点击绿旗执行程序后&#xff0c;下列说法正确的是&#xff1f;&#xff08;C&#xff09;&#xff08;2分&#xff09; A.小猫一直在左右移…

JDBC详解(二):获取数据库连接(超详解)

JDBC详解&#xff08;二&#xff09;&#xff1a;获取数据库连接 前言一、要素一&#xff1a;Driver接口实现类1、Driver接口介绍2、加载与注册JDBC驱动 二、要素二&#xff1a;URL三、要素三&#xff1a;用户名和密码四、数据库连接方式举例4.1 连接方式一4.2 连接方式二4.3 连…

15天学习MySQL计划-数据库引擎(进阶篇)第六天

15天学习MySQL计划-数据库引擎&#xff08;进阶篇&#xff09;第六天 1.数据库引擎 1.MySQL体系结构 连接层服务层引擎层存储层 2.存储引擎 存储引擎简介 ​ 1.概述 ​ 存储引擎就是存储数据&#xff0c;建立索引&#xff0c;更新/查询数据等技术的实现方式。存储引擎是基…

android ANativeWindow surface显示

前言 最近做车机camera 倒车影像问题&#xff0c;需要通过c调用camera&#xff0c;并显示在android ui界面之上。 最终效果图 代码实现 Android.bp cc_binary {name: "stest",vendor: true,srcs: ["main.cpp"],shared_libs: ["libcutils",&q…

Android入门基础教程

第1章 Android Studio运行第一个程序 1.1 Android Studio下载&#xff1a; 1.1.1 Android开发者官网&#xff1a; https://developer.android.google.cn ​ 1.1.2 下载Android Studio开发者工具&#xff1a; 进入Android开发者官网&#xff1b;找到Android Studio工具下载页…

【LeetCode刷题笔记】反转链表、移除链表元素、两两交换链表中的节点、删除链表的倒数第N个结点

&#x1f4dd;个人主页&#xff1a;爱吃炫迈 &#x1f48c;系列专栏&#xff1a;数据结构与算法 &#x1f9d1;‍&#x1f4bb;座右铭&#xff1a;道阻且长&#xff0c;行则将至&#x1f497; 文章目录 反转链表移除链表元素两两交换链表中的节点删除链表的倒数第 N 个结点&…