Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >扒一扒这个注解,我发现还有点意思。

扒一扒这个注解,我发现还有点意思。

作者头像
why技术
发布于 2021-12-27 11:17:52
发布于 2021-12-27 11:17:52
6130
举报
文章被收录于专栏:why技术why技术

你好呀,我是歪歪。

我上周我不是发了这篇文章嘛,《我也想说说日志,但是我不想说漏洞。》

然后就有小伙伴来问我:我项目里面用的是 Lombok 的 @Slf4j 这个会有影响吗?

你说这事多巧,我也用的这个注解,所以我当时稍微的看了一下。

先说结论:有没有影响还是取决于你项目中依赖的 log4j2 包,和 Lombok 没有任何关系。

另外“求求你提问之前三思,不要浪费我们的时间,不要问那些你自己就能搞清楚的问题”这句话不是我说的,是 Lombok 的作者说的:

他为什么会说出这样的略带一丝丝气愤的话呢?

我带你看看。

从issue 说起

你在 github 上找到 Lombok 项目,然后查看它的 issue 会看到关于 log4j 的问题已经被置顶了:

所以关于这个问题,我就从这个 issue 说起吧,这里面有 Lombok 维护者的权威回答。

https://github.com/projectlombok/lombok/issues/3063

这个 issue 的标题翻译过来就是:

结论:关于 log4j 的 0day 问题,Lombok 本身不受影响。

注意啊,这个里面有个非常耐人寻味的单词:itself。

一般我们说 Lombok 不受影响就行了,为什么还得加个 itself(本身)呢?

这里面就是有故事的。

首先是 12 月 10 日下午 5 点 41 分,也就是漏洞被爆出的那个下午,有个哥们在 Lombok 的 issue 提出了这个问题:

他说:铁子们,出事了,log4j 爆出惊天大漏洞,赶紧补吧。

然后有个叫做 Rawi01 的老铁跳出来说:

老哥,你能解释一下这个漏洞 对 Lombok 有什么影响吗?据我所知,log4j 只是在运行测试时不出现编译错误时才需要。

这个老哥明显是对 Lombok 比较了解的,他其实已经指出关键的地方了:Lombok 并没有依赖 log4j。

但是紧接着有人指出:

Lombok 提供了一个注解,叫做 @log4j。这玩意使用了 Log4J 的库吧,毕竟要用它记录日志。

确实有这个注解啊,给大家看一下:

我一般是用 @Slf4j 这个注解,但是它确实是还提供了其他的日志注解,这个先按下不表,等下再说。

现在的主线任务是这个 issue,接着往下看。

接下来,来了一个控场的哥们,看一下他说了啥:

他先艾特了前面的说 @log4j 注解的这个哥们,给他解释说:虽然 Lombok 会生成代码来创建一个 logger 的实例,但我们并没有 ship, distribute or require 任何特定的版本,如果用户要用这个注解,需要他们自己提供对应的依赖。

这里我有三个词我还没翻译:ship, distribute or require。

request 其实很简单,就是要求。

意思是我们没有要求用户使用的时候必须给到指定版本的依赖。说白了就是,假设你要用 @Log4j,那就提供对应的依赖,至于这个依赖的版本是什么 Lombok 并不关心。

剩下的 ship 直译过来是“运送” 的意思,而 distribute 有“分发”的意思。

这个我用一张图片来解释吧:

可以看到,在项目中我引用了 Lombok,但是它并没有传递依赖进来任何其他的包。不像是 spring-boot-starter 后面跟着一大坨东西。

我理解这就是他想要表达的:we do not ship, distribute or require any specific version.

然后接着说:在我们的代码库中确实可以找到一个 log4j2 的版本,但那只是在测试代码里面使用,以便能够无错误地编译生成的代码。

最后这个哥们总结到:放心,老铁们,Lombok 仍然可以安全的使用。但我觉得虽然是测试代码中引用到了 Log4j ,也还是应该把依赖关系更新到安全的版本。

怎么样,是不是感觉这个老哥说话有一种不可置疑的自信感。

我读完这个回答之后就是有这样的感觉,隐隐觉得这就是个大佬。

于是我去扒了一下这个老哥的背景:

本来我是想从他的 github 主页上寻找答案,居然没有发现和 Lombok 相关的项目。

而且只有 49 个 followers,看着也不符合大佬的数据呀:

于是我转战浏览器,搜索了:Roel Spilker Lombok.

找到了这个:

才发现,好家伙,他是 Lombok 的爸爸啊,难怪说话这么硬气。

还发现了一个花边新闻:他们之前还把关于 Lombok 的想法贡献给了 Oracle,但是被拒绝了。官方不支持,他们只有揭竿而起,自己搞了。

现在 Lombok 的市场占有率还是很高的吧,算是把这事儿干成功了。

另外,其实这里就可以直接看到他的身份:

Collaborator,就是合作人。他的回答可以等同于官方回答了。

到这里事情还算是发展的比较顺利,官方亲自下场来解答这个问题,且已经明说了:

算是可以完结撒花了。

但是,就怕出现但是啊,后面发生的事情我就觉得有点离谱了。

首先是 SunriseChair 老哥献出了一杀:

他首先引用了作者的回复,然后说,如果你在你的 maven 或者 gradle 中声明 Lombok 这个依赖项,那么 Log4j 的依赖项不也会被包括在内吗?这可能是 classpath 上唯一的依赖,而 Lombok 生成的代码也会用到它?如果我说错了,请纠正我......

他想要表达的是什么意思呢。

首先我觉得是他读完作者的回答后觉得 Lombok 是依赖了 Log4j 的,所以他的核心问题就是,如果我引用了 Lombok 那么 Log4j 的依赖岂不是也会传递进来?

但是作者都说了:there is a version to be found in our code base, but that's 仅仅用于测试类中。

但凡你稍微了解一点开源项目的工作原理,也就知道测试相关的部分都不会提供出去,如果你要看测试类,你得把项目源码拉下来。

简单来说就是如果你通过 maven 依赖了 Lombok,测试相关的东西你肯定是看不到的。

接着一位叫做 RuanNunes 的老哥也出来补枪了,献上了二杀,他说他在整个项目里面搜索到了这个东西:

这个配置是存在漏洞的。

然后作者对这两个问题进行了一一回复:

首先,说 Lombok 依赖了 log4j 的老哥听着:Lombok 没有对 log4j 或任何其他库进行依赖。如果你在你的代码中使用 @Log4j 注解,但是又不直接或间接地依赖 log4j ,你的编译就会产生一个错误信息。

关于这一点,前面的依赖分析截图也已经说明了,我就不再贴一次了。

然后说找到漏洞的哥们听着:在你找到的这个漏洞的同一目录下有一个文本文件,但凡你去看一眼,你就知道我们不会将该文件作为我们发布的一部分。

所以我理解压根不会发布出去,即使有漏洞,对于 Lombok 的用户来说也没有任何毛病吧?

我也去找了一下作者说的这个文件,就是它:

https://github.com/projectlombok/lombok/blob/d3b763f9dab4a46e88ff10bc2132fb6f12fda639/src/support/info.txt

这个文件里面清清楚楚的说了:not part of lombok itself.

如果前面这两个问题,已经让作者有点点不爽的感觉的话,那么接下来这个问题,可能就是引爆点,完成三杀,一波带走:

这个哥们上来就说:老铁,我问个问题哈。我使用的是 @slf4j 注解,这个漏洞和我有关系吗?

不行了不行了,血压上来了。

我觉得这个哥们提问之前完全没有看前面的回复,如果他在作者进行了两次解释之前提出这个问题,那么我觉得完全可以理解。

但是在作者已经在这个 issue 里面解释了两次“Lombok 不受影响”的前提下,他哗一下,上来就是一个暴击:

老哥,我用的这个注解有问题吗?

我写到这里都有点血压上来了。

这特么特别像是之前有一次在一个群里和别人讨论事务失效的场景,已经讨论到尾声了,突然跳出来一个哥们,甩出来一个代码片段,这个片段的写法就是一个经典的 this 调用导致事务注解失效的场景,而这就是我们刚刚讨论中的一部分。

知道我们在讨论这个问题,哪怕你稍微往上翻一翻呢,了解一下上下文再提问不好吗?

不扯远了,也许是我过度解读把,反正我觉得提问的艺术大家都得好好学一下。

回到我们的主线剧情,看一下作者是怎么回答这个问题的:

他说:老铁,这个问题我帮不了你啊。我对于这个漏洞一无所知,我甚至不明白为什么你会认为它也会影响 @Slf4j。

我可以告诉你的是,Lombok 没有使用、传递、要求对这些库的依赖。

我们的工作原理是生成了你"看不见的源代码"。

如果你有任何理由怀疑类似的问题也可能发生在他们的产品中,请联系 Slf4j 的维护者。

不知道是不是我个人的感觉,我觉得作者回复这个问题的时候已经有一点怒气了:这都是些什么问题啊?前我已经回复了两次不受影响啊?怎么感觉我的项目使用者怎么都不了解 Lombok 的基本原理呢?

在作者回复完这个冒失的提问者之后,这个提问者还是很礼貌的回复了一下:

谢了老铁,我检查了,我使用的是 SpringBoot 默认的 logback。

作者也把这个问题给置顶了,且修改了这个 issue 的标题。

所以,经过前面的一番解读,现在你再看这个标题:

为什么作者要强调“itself”,因为 Lombok 确实是提供了日志的功能,但是至于引用什么包,哪个版本的包,和 Lombok 都没有任何关系。Lombok 本身(itself)是安全的。

最最后,作者给了 log4j 漏洞对于 Lombok 的 Latest assesment(最新评估),算是总结性发言:

给大家翻译一下关键的东西。

这个漏洞只存在于 2.16.0 版本以下的 Log4j code 包中,而不存在于任何其他日志框架中。

Lombok 没有传递依赖任何 Log4j 包,也没有声明对任何东西的依赖。

如果你使用任何 Lombok 的注解,比如 @Log4j,Lombok 将生成使用这些库的代码,但是你的项目里面必须要包含对这些库的依赖,否则 Lombok 生成的代码将无法编译。

同样地,你要负责在你的运行时中拥有这些包,否则类的初始化可能会失败。

在 Lombok 测试代码中,我们曾经有一个包含这个漏洞的版本,但是由于测试不处理任何用户输入(测试是硬编码的),而且生成的代码甚至没有被执行,运行测试并没有导致执行测试的机器上出现 RCE(远程代码/命令执行漏洞)。

所以,老铁们,Lombok 本身不需要做任何改变,也不对你的项目负任何安全责任,毕竟包不是我们引进来的。

如果你不同意目前的评估,请在这个问题上添加评论。

但是,请确保你已经阅读了其他评论,并确保你理解了这个问题。

最后这两句话,单独拎出来,我可太喜欢这两句话了:

求求你提问之前三思,不要浪费我们的时间,不要问那些你自己就能搞清楚的问题。

如果你认为我们遗漏了什么,或者有新的信息,请大声的说出来。

然后,你注意这里作者用的小标题是:The balancing act.

翻译过来是“平衡的行为”,啥玩意?

NO,NO,NO:

一个小俚语送给大家,不必客气。

补充说明

前面把主线剧情过完了,现在我来几个补充说明吧。

先说前面按下不表的关于日志的注解。

其实 Lombok 里面关于日志的注解还真是挺多的,可以直接看官方的文档:

https://projectlombok.org/features/log

这么多注解,一个个的讲也没啥意思,我这里就挑 @Slf4j 、@Log4j2 这两个演示一下吧。

首先,我们可以搞个纯净的 SpringBoot 项目,只包含这两个依赖:

这个时候如果我什么都不动,只是稍微改一下启动类:

然后为了排除干扰项我把日志打印的级别调整到 Error:

logging.level.root=error

同时关闭 banner 输出:

spring.main.banner-mode=off

banner 就是这个玩意:

这个时候启动项目,日志输出是这样的:

可以看到我们这个时候使用的日志是 logback,原因我在之前的文章里面也讲过了,因为 Springboot 默认使用的日志实现是 logback。

这一点,从项目依赖上也可以看出来:

另外,你注意一下,我特意把 import 部分也截出来了,除了 @Slf4j 注解外,这里并没有引入任何日志相关的注解。

然后,我再关注一下这个时候编译出来的 class 文件:

自动引入了 slf4j 相关的包,然后生成了这行代码:

private static final Logger log = LoggerFactory.getLogger(LogdemoApplication.class);

这个时候不知道你有没有想到编译时注解相关的东西,但是不慌,这里还是先按下不表。

来,我问你:为什么它能引入 slf4j 相关的包?

因为我依赖了呀:

好,如果这个时候我把 logback 的核心依赖给拿掉,会出现什么事情,你觉得会不会编译不过呢?

不会编译不过,因为 Slf4j 包还在,它只是一个日志门面。

但是运行的时候会抛出异常,因为找不到日志相关的具体实现类:

然后,如果我想用 log4j2 日志实现怎么办呢?

之前的文章中也写过:

把 Springboot 默认的依赖拿掉,然后引入 Log4j2 的包。

这个时候项目依赖图是这样的,可以看到没有 logback-core 了,只有 log4j-core:

再次运行项目,日志实现就变成了 Log4j:

你发现了吗,我除了动了一下 pom 依赖外,其他的代码一行都没有动,日志框架就从 logback 变化为了 log4j。

而且 class 文件没有任何变化,所以我也就不去截图了。

这就是 Slf4j 的功劳,这就是“门面”的含义,这就是为什么都建议大家在项目中使用 Slf4j,而不是具体的诸如 logback、log4j 这样具体的日志实现。

接下来说一下 @Log4j2 这个注解。

我们还是把依赖恢复到最开始纯净的状态,也就是这样:

然后我们把注解修改为 @Log4j2,但是我们项目中这个时候并没有引入 Log4j-core 包,那么你觉得会有问题吗?

不会有问题的,我们可以看一下。

先看一下输出:

此时的日志实现类是 SLF4JLogger。

这玩意哪里来的?

看一下 class 文件:

这个两个类是来自于 log4j-api 包里面,同时由于 log4j-to-slf4j 包的存在,所以最后的实现类桥接到了 SLF4JLogger 中去:

如果我把 log4j-api 包移除掉,你说会不会编译不过呢?

肯定编译不过的,因为包都不存在了,搞不出来 class 文件:

如果我不想用 SLF4JLogger 这个类呢,我就想用真正的 log4j。

简单,把 log4j 的依赖搞进来:

好,我前面说了这么多的废话,不厌其烦的给你排除、引入日志相关的包,给你看输出啥的,而且整个过程中并不涉及到 Lombok 包的变化,都是为了再次印证这两句话:

如果你使用任何 Lombok 的注解,比如 @Log4j,Lombok 将生成使用这些库的代码,但是你的项目里面必须要包含对这些库的依赖,否则 Lombok 生成的代码将无法编译。

比如我前面把 log4j-api 包移除掉了,是不是编译就没有过?

同样地,你要负责在你的运行时中拥有这些包,否则类的初始化可能会失败。

比如我前面把 logback-core 的包移除了,编译的时候没有问题,但是服务运行的时候,是不是抛出找不到类的异常?

是不是再次证明:

聊聊原理

前面我提到了一句“编译时注解”,不知道大家对于这个玩意了不了解。

不了解其实也很正常,因为我们写业务代码的时候很少自定义编译时注解,顶天了搞个运行时注解就差不多了。

Lombok 的核心工作原理就是编译时注解。

其实我了解的也不算深入,只是大概知道它的工作原理是什么样的,对于源码没有深入研究。

但是我可以给你分享一下两个需要注意的地方和可以去哪里了解这个玩意。

首先第一个需要注意的地方是这里:

log 相关注解的源码位于这个部分,可以看到很奇怪啊,这些文件是以 SCL.lombok 结尾的,这是什么玩意?

这是 lombok 的小心思,其实这些都是 class 文件,但是为了避免污染用户项目,它做了特殊处理。

所以你打开这类文件的时候选择以 class 文件的形式打开就行了,就可以看到里面的具体内容。

比如你可以看看这个文件:

lombok.core.handlers.LoggingFramework

你会发现你们就像是枚举似的,写了很多日志的实现:

这个里面把每个注解需要生成的 log 都硬编码好了。正是因为这样,Lombok 才知道你用什么日志注解,应该给你生成什么样的 log。

比如 log4j 是这样的:

private static final org.apache.logging.log4j.Logger log = org.apache.logging.log4j.LogManager.getLogger(TargetType.class);

这也同时可以和我们前面的 class 文件对应起来:

而 SLF4J 是这样的:

private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(TargetType.class);

第二个需要注意的地方是找到入口:

这些 class 文件加载的入口在于这个地方,是基于 Java 的 SPI 机制:

AnnotationProcessorHider 这个类里面有两行静态内部类,我们看其中一个, AnnotationProcessor ,它是继承自 AbstractProcessor 抽象类:

javax.annotation.processing.AbstractProcessor

这个抽象类,就是入口中的入口,核心中的核心。

在这个入口里面,初始化了一个类加载器,叫做 ShadowClassLoader:

它干的事儿就是加载那些被标记为 SCL.lombok 的 class 文件。

然后我是怎么知道 Lombok 是基于编译时注解的呢?

其实这玩意在我看过的两本书里面都有写,有点模糊的印象,写文章的时候我又翻出来读了一遍。

首先是《深入理解 Java 虚拟机(第三版)》的第四部分程序编译与代码优化的第 10 章:前端编译与优化一节。

里面专门有一小节,说插入式注解的:

Lombok 的主要工作地盘,就在 javac 编译的过程中。

在书中的 361 页,提到了编译过程的几个阶段。

从Java代码的总体结构来看,编译过程大致可以分为一个准备过程和三个处理过程:

  • 1.准备过程:初始化插入式注解处理器。
  • 2.解析与填充符号表过程,包括:
    • 词法、语法分析。将源代码的字符流转变为标记集合,构造出抽象语法树。
    • 填充符号表。产生符号地址和符号信息。
  • 3.插入式注解处理器的注解处理过程:插入式注解处理器的执行阶段,本章的实战部分会设计一个插入式注解处理器来影响Javac的编译行为。
  • 4.分析与字节码生成过程,包括:
    • 标注检查。对语法的静态信息进行检查。
    • 数据流及控制流分析。对程序动态运行过程进行检查。
    • 解语法糖。将简化代码编写的语法糖还原为原有的形式。(java中的语法糖包括泛型、变长参数、自动装拆箱、遍历循环foreach等,JVM运行时并不支持这些语法,所以在编译阶段需要还原。)
    • 字节码生成。将前面各个步骤所生成的信息转换成字节码。

如果说 javac 编译的过程就是 Lombok 的工作地盘,那么其中的“插入式注解处理器的注解处理过程”就是它的工位了。

书中也提到了 Lombok 的工作原理:

第二本书是《深入理解 JVM 字节码》,在它的第 8 章,也详细的描述了插件化注解的处理原理,其中也提到了 Lombok:

最后画了一个示意图,是这样的:

如果你看懂了书中的前面的十几页的描述,那么看这个图就会比较清晰了。

总之,Lombok 的核心原理就是在编译期对于 class 文件的魔改,帮你生成了很多代码。

这也是作者提到的:

invisible source code,看不见的源码。

这里的看不见,指的是 java 文件中的看不见,在 class 文件中它还是无处遁形。

如果你有兴趣深入了解它的原理的话,可以去看看我前面提到的这两本书,里面都有手把手的实践开发。

我就不写了,一个原因是因为确实门槛较高,写出来生涩难懂。另外,一个原因那不是因为我懒嘛。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-12-20,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 why技术 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
日志框架与门面模式
第三方的日志框架除了提供具体的日志实现外,也有的会提供自身框架的接口。此外,有些日志框架仅仅提供了一套统一的接口,不包含具体的日志实现,可以通过一些桥接包来调用其他的日志框架。(即日志的门面模式)
雨临Lewis
2022/01/12
1.1K0
基础框架的那些事
diary_report.jpg What did you do today 日志门面有:jcl(Apache Commons Logging), slf4j, jboss-logging(不是服务大众的)。日志实现有:log4j, log4j2,logback,jul(java.util.logging)。公司的日志框架选型是slf4j + log4j2。现在捋一捋他们之间的关系,进行jar包总结。 log4j1 : log4j.jar有log4j的全部内容。 log4j2:log4j-api包
用户2032165
2018/06/05
9220
带你深入Java Log框架,彻底搞懂Log4J、Log4J2、LogBack,SLF4J
使用过Log4J和LogBack的同学肯定能发现,这两个框架的设计理念极为相似,使用方法也如出一辙。其实这个两个框架的作者都是一个人,Ceki Gülcü,俄罗斯程序员。
码老思
2023/10/19
5.2K0
带你深入Java Log框架,彻底搞懂Log4J、Log4J2、LogBack,SLF4J
Java日志体系框架总结:JUL、JCL、SLF4J、Log4j、Logback、Log4j2
日志记录是应用程序运行中必不可少的一部分。具有良好格式和完备信息的日志,可以在程序出现问题时帮助开发人员迅速地定位错误的根源。日志所能提供的功能是多种多样的,包括记录程序运行时产生的错误信息、状态信息、调试信息和执行时间信息等。
johnny666
2024/09/24
4630
SLF4J框架理解与分析
这两天在看设计模式相关的书,正好看到了门面模式,感觉不太能领悟它的精髓,就想找一些例子来看,突然发现这个slf4j框架不就是一个门面(facade /fəˈsɑ:d/)么,干脆就直接拿来看一看了,正好也把java的日志系统也了解了解。
mythsman
2022/11/14
5400
Slf4j适配日志原理
看了之前的文章Java日志体系总结后,相信大家对slf4j以及其他日志组件的关系有了一定理解。slf4j只是为日志的输出提供了统一接口,并没有具体的实现,就好像JDBC一样。那么,大家会不会好奇slf4j是怎么绑定/适配/桥接到log4j或者logback其他日志实现组件的呢?这篇文章为大家详细讲述。
Erwin
2020/07/30
1.1K0
Slf4j适配日志原理
log4j2如何实现日志文件的生命周期管理
对于日志的认知,不同阶段是不一样的。在大学刚学Java的时候,根本不理解日志的用处,甚至觉得日志和控制台输出的内容不一样吗。后来学到log4j,通过配置输出INFO、ERROR不同等级的日志,才明白,原来日志和控制台输出是不一样。
叫我阿柒啊
2025/01/17
2300
log4j2如何实现日志文件的生命周期管理
【JAVA】Java 日志打印规范
规范的日志是养成良好编程习惯的开始,也是关键时刻解决严重BUG的救命稻草。程序员开发的过程中可以打印debug日志,在复杂业务中提供日志来排查问题,也可以在出现生产问题的时候快速问题,及时处理。无论如何了解和学习日志的规范是程序员必备的基本功。
阿东
2022/12/30
5.8K0
【JAVA】Java 日志打印规范
Java日志框架学习--LogBack和Log4j2--下
Logback当前分成三个模块:logback-core,logback- classic和logback-access。
大忽悠爱学习
2022/05/17
1.3K0
Java日志框架学习--LogBack和Log4j2--下
【SpringBoot专题】Java平台下日志的那些事前言日志框架漫谈看SpringBoot如何对日志进行统一处理SpringBoot日志使用结束语
本篇是【SpringBoot专题】系列的第三篇,将介绍SpringBoot对日志的支持,讲解Java平台下日志的那些事,彻底揭开日志框架在使用过程中的那些坑~
用户2890438
2018/08/21
6050
【SpringBoot专题】Java平台下日志的那些事前言日志框架漫谈看SpringBoot如何对日志进行统一处理SpringBoot日志使用结束语
你好,SLF4J
作为 CV 工程师,咱们开发的应用并不总是按预期运行,为了方便排查出潜在的问题,一般会在代码中添加日志记录语句。但在 Java 刚刚问世时,日志记录方式好像除了System.out和System.err之外也没啥别的选择了,主要痛点有:1) 日志无法分级,有些日志纯属 DEBUG,在生产环境是不需要的;2) 日志内容不支持格式化,如 XML、HTML。后来,一位名叫Ceki Gülcü的大神无奈之下发布了大名鼎鼎的log4j。尽管现在 log4j 逐渐退出历史舞台,但在当时却备受 Java 开发人员的喜爱,甚至 JDK 1.4 也是借鉴了 log4j 之后,终于在官方类库中补齐了日志记录这一短板,它就是j.u.l包。
程序猿杜小头
2022/12/01
7730
你好,SLF4J
芋道 Spring Boot 消除冗余代码 Lombok 入门
摘要: 原创出处 http://www.iocoder.cn/Spring-Boot/Lombok/ 「芋道源码」欢迎转载,保留摘要,谢谢!
芋道源码
2020/04/01
1.1K0
全网最全、最细致的Java日志框架以及门面技术。
👨‍🎓作者:Java学术趴 🏦仓库:Github、Gitee ✏️博客:CSDN、掘金、InfoQ、云+社区 💌公众号:Java学术趴 🚫特别声明:原创不易,未经授权不得转载或抄袭,如需转载可联系小编授权。 🙏版权声明:文章里的部分文字或者图片来自于互联网以及百度百科,如有侵权请尽快联系小编。微信搜索公众号Java学术趴联系小编。 ☠️每日毒鸡汤:生活是世上最罕见的事情,大多数人只是存在,仅此而已。 👋大家好!我是你们的老朋友Java学术趴,今天继续给大家分享小趴Java日志框架。对于一个应用程序来说
Java学术趴
2021/09/10
4.2K0
web项目中如何选择日志组件(SLF4J、Log4J2、logback)
    为什么使用SLF4J与其他组件结合使用比单独使用log4j2或者java.util.logging要优秀呢。
洋仔聊编程
2019/01/15
4.8K0
Apache Log4j2详解
在项目开发中,都不可避免的使用到日志。没有日志虽然不会影响项目的正确运行,但是没有日志的项目可以说是Java
Java架构师必看
2021/07/15
1.3K0
深入理解jar包冲突的本质
上篇文章 记一次log4j不打印日志的踩坑记 介绍了遇到的log4j踩坑经历和解决方法,这篇文章我们重点来学习和了解下有关Java中日志组件的内容,在这之前,其实在我的头脑里,并没有形成系统的日志框架知识,原因其实是一直没有重视过这块,之前都是各种拷贝改改能跑就行,并不理解相关的架构和原理,这次趁着这个机会正好来系统了解一下,除了要系统的理解日志框架大多数知识外,我们还要学习一个非常关键的知识,就是关于Java默认的类加载器加载jar包的顺序问题,不夸张的说,只有理解了这个,才能搞明白jar冲突问题发生的本质。
我是攻城师
2019/09/25
2.2K0
深入理解jar包冲突的本质
教你全方位解决Java 日志框架冲突!
你是否遇到过配置了日志,但打印不出来的情况?你是否遇到过配置了 logback,启动时却提示 log4j 错误的情况?像下面这样:
二哥聊运营工具
2022/07/11
1.1K0
教你全方位解决Java 日志框架冲突!
slf4j、log4j、log4j2、logback到底用哪些jar
SparkStreaming用久了,打算学习一下Flink,就从官网下载了Flink 1.11,打算搞一个客户端,将程序提交在yarn上。因为Flink从1.7之后就不再提供Hadoop的依赖,所以很多依赖就要自己下载,于是各种ClassNotFoundException,其中以log*.class为首的格外猖狂,可能是因为flink和Hadoop的日志实现有点区别,就一直哐哐哐报错,slf4j、log4j、logback各种jar包十几个,百度好久也没搞清各个jar有什么区别,用在何处,就打算自己总结一下。
叫我阿柒啊
2022/05/09
4340
slf4j、log4j、log4j2、logback到底用哪些jar
JAVA日志框架适配/冲突解决方案
上面的这些问题,基本都是由于多套日志框架共存或配置错误导致的。那么为什么会出现共存或者冲突呢? 一般是以下几种原因:
神秘的寇先森
2021/04/13
1.6K0
忽视日志吃大亏,手把手教你玩转 SpringBoot 日志
对于刚学习编程的同学,很多人都对日志满不在乎,我们在做code review的时候,经常发现一些新同学喜欢一个方法写得很长,然后中间的注释和日志都少的可怜。
程序员大彬
2023/11/03
4.5K1
忽视日志吃大亏,手把手教你玩转 SpringBoot 日志
相关推荐
日志框架与门面模式
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档