Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >日志那些事儿——由一次bug引发的思考-client jar应该如何输出日志

日志那些事儿——由一次bug引发的思考-client jar应该如何输出日志

作者头像
LNAmp
发布于 2018-09-05 07:44:53
发布于 2018-09-05 07:44:53
5090
举报

前言

前面几篇“日志那些事儿”讲解了日志的重要性和相关使用。以slf4j+logback的使用为例,我们的步骤为:

  • 在工程中引入slf4j、logback相关Jar包
  • 编写配置文件logback.groovy/logback.xml等
  • 使用LoggerFactory.getLogger()得到logger使用。 从上面可以看出使用过程中非常重要的一个部分为编写配置文件logback.xml,配置文件中通常是配置我们所需要的appender和相关的logger,例如说配置console输出或者配置文件输出等。

一次需求和一次bug

一次需求

前段时间做过一次需求,要求的是尽量无侵入的输出某些请求参数到日志中,Web框架是Webx,在web层采用了valve实现了类似切片功能,在service层直接使用了aop。当然这些都不算关键,关键在于要尽量少的侵入应用,并且要在多个系统中使用。由于在不同应用中可能使用了不同类型的日志框架,所以将关键的aop逻辑、日志输出逻辑封装在client jar中,供应用系统使用。

一次bug

起因

前两天需要对系统升级某项功能,需要引入一个新的jar,升级完成在日常测试环境中部署够,服务功能正常,但是却发现一个非常严重的bug——所有的日志全部不输出了。经排查之后发现是引入的新client jar中使用了slf4j+log4j2,而原应用中使用的是slf4j+logback。前者中引入log4j2-slf4j-impl,后者中的logback-classic中都用org.slf4j.impl.StaticLoggerBinder.class,所以会发生multiple_bindings的错误,按照slf4j的官方文档的说明:

NOTE The warning emitted by SLF4J is just that, a warning. Even when multiple bindings are present, SLF4J will pick one logging framework/implementation and bind with it. The way SLF4J picks a binding is determined by the JVM and for all practical purposes should be considered random. As of version 1.6.6, SLF4J will name the framework/implementation class it is actually bound to.

由于两个StaticLoggerBinder.class属于同名同包,按照hotspot的Jvm规范,会加载classpath下靠前的那个jar中的StaticLoggerBinder。

If the 2 classes have the same package name, i.e. com.mycompany.Client, then you end up in a situation where it is somewhat arbitrary which version of Client is loaded. It comes down to which is on the classpath first. This is a JAR hell situation.

所以,日志停止输出的原因很肯定是因为slf4j binding到了log4j-slf4j-impl中的StaticLoggerBinder,而没有使用logback进行输出。

解决

分析了原因,解决起来了就很简单了,暴力exclude了引入的client jar包中的log4j2相关依赖。重新部署验证后发现日志可以输出,问题解决

对于slf4j的思考
  • 这样做虽然解决了问题,但是并不完美。因为client jar包中引入了log4j2,并且配置了log4j2.xml的配置文件,本意肯定是想使用log4j2输出相关日志到自己指定文件,但是由于我暴力干掉了log4j2,client想输出日志到指定文件是不可能的了,client会将其日志输出到应用主日志中。
  • 当系统中出现两个及以上StaticLoggerBinder.class的时候,肯定不算是“最佳实践”。日志系统虽多,择其善者而从之。

对于client jar设计的思考

一次需求,一次bug,其实都与日志系统相关。这次bug的根本原因也算是因为client jar没有考虑到使用者的感受,把log4j2强加给使用者。client jar可能会有一些记录日志的需求,例如记录一些信息用于调试。那么当client jar有日志输出需求,如何更好地进行设计呢?

我觉得得从以下几个方面考虑:

  • 缩小jar包依赖范围,如果是基于maven,可以把client工程中依赖的日志相关jar scope设置为provided.
  • “感知”应用系统所使用的日志框架,匹配相应的日志框架
  • 不使用配置文件,改用编码配置logger

对于第二点,client jar必须要“感知”应用系统所使用的日志框架,例如究竟是使用了log4j还是log4j2,还是slf4j+logback,亦或者slf4j+log4j等,这样才可以知道client jar中可以使用哪些日志api。对于第三点,由于我们一般都通过xml等配置文件配置日志logger,但这种方式很不灵活。要适配多少种日志系统,就需要在client jar的classpath下放置多少种日志配置文件,而且没法在运行时指定appender,设置layout等等。

总结

如果在client jar中有日志输出的需求,一定要好好设计,千万不能坑了使用者。下篇文章将围绕如何设计包含日志输出的client jar。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2016.09.03 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

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

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

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Springboot 系列(四)Spring Boot 日志框架
Spring 框架选择使用了 JCL 作为默认日志输出。而 Spring Boot 默认选择了 SLF4J 结合 LogBack。那我们在项目中该使用哪种日志框架呢?在对于不同的第三方 jar 使用了不同的日志框架的时候,我们该怎么处理呢?
不会飞的小鸟
2020/06/07
3.5K0
日志那些事儿——谈谈需要日志输出的client jar应该如何设计
上篇文章提到了应该如何设计需要输出日志的client jar,大概有三个比较重要的点。
LNAmp
2018/09/05
6410
日志那些事儿——谈谈需要日志输出的client jar应该如何设计
[享学Feign] 十一、Feign通过feign-slf4j模块整合logback记录日志
代码下载地址:https://github.com/f641385712/feign-learning
YourBatman
2020/02/21
5.1K0
[享学Feign] 十一、Feign通过feign-slf4j模块整合logback记录日志
排查log4j不输出日志到文件的问题
但是在项目代码中输出的日志信息始终不输出到文件中,只在控制台输出。 一开始我以为是log4j的配置问题:只输出到控制台,不输出到文件,但是反复确认配置没问题。
编程随笔
2022/04/29
3.5K0
深入浅出日志体系(logback最佳实践)
今天跟大家分享一篇我比较敬佩的业界大佬—张建飞同学(阿里高级技术专家,cola框架作者)刚新鲜出炉的一篇关于日志实践的文章。大致分享的是如下几点内容:
陶朱公Boy
2023/01/07
1.1K0
深入浅出日志体系(logback最佳实践)
【SpringBoot专题】Java平台下日志的那些事前言日志框架漫谈看SpringBoot如何对日志进行统一处理SpringBoot日志使用结束语
本篇是【SpringBoot专题】系列的第三篇,将介绍SpringBoot对日志的支持,讲解Java平台下日志的那些事,彻底揭开日志框架在使用过程中的那些坑~
用户2890438
2018/08/21
5780
【SpringBoot专题】Java平台下日志的那些事前言日志框架漫谈看SpringBoot如何对日志进行统一处理SpringBoot日志使用结束语
带你深入Java Log框架,彻底搞懂Log4J、Log4J2、LogBack,SLF4J
使用过Log4J和LogBack的同学肯定能发现,这两个框架的设计理念极为相似,使用方法也如出一辙。其实这个两个框架的作者都是一个人,Ceki Gülcü,俄罗斯程序员。
码老思
2023/10/19
3.9K0
带你深入Java Log框架,彻底搞懂Log4J、Log4J2、LogBack,SLF4J
Java日志体系权威总结
本文的目的是搞清楚Java中各种日志Log之间是怎么的关系,如何作用、依赖,好让我们平时在工作中如果遇到“日志打不出”或者“日志jar包冲突”等之类的问题知道该如何入手解决,以及在各种场景下如何调整项目中的各个框架的日志输出,使得输出统一。
程序猿DD
2020/12/18
5560
Java日志体系权威总结
SpringBoot 笔记 ( 三 ):日志系统
SpringBoot 笔记 ( 三 ):日志系统 1、日志框架 日志框架就是防止我们再去像以前那样,一直进行System.out.println(“”)将关键数据打印在控制台。框架来记录系统的一些运行时信息,但是随着日志框架的增长,和接口的不一致,导致了使用上的差别很大,​这里采用了一个类似于数据库驱动的模式,数据库驱动是 Java 提供的一个 API,然后真正的实现是需要各个数据库厂商去完成的,而 log 也开始采用这种面向接口编程的方法采用日志抽象层。 市面上的日志框架 JUL、JCL、Jboss-l
lwen
2018/04/16
2.1K0
LogBack入门实践
slf4j是The Simple Logging Facade for Java的简称,是一个简单日志门面抽象框架,它本身只提供了日志Facade API和一个简单的日志类实现,一般常配合Log4j,LogBack,java.util.logging使用。Slf4j作为应用层的Log接入时,程序可以根据实际应用场景动态调整底层的日志实现框架(Log4j/LogBack/JdkLog...);
zhangheng
2020/04/28
5720
java日志commons-logging/log4j/slf4j/logBack需要知道的几件事
如果对于commons-loging、log4j、slf4j、LogBack等都已经非常清楚了,可以忽略本文。几次解决日志冲突问题时对这几个概念的简单总结,希望对这块基础没有理解透的同学能有所帮助,当然如果对这块有更深刻理解的同学,也贡献出自己的知识和见解。
一个会写诗的程序员
2018/08/17
7470
java日志commons-logging/log4j/slf4j/logBack需要知道的几件事
Java日志体系(slf4j)
3 slf4j 3.1 简介 与commons-logging相同,slf4j也是一个通用的日志接口,在程序中与其他日志框架结合使用,并对外提供服务。 Simple Logging Facade fo
贾博岩
2018/05/11
5.3K0
SpringBoot整合日志框架
​ 1、System.out.println("");将关键数据打印在控制台;去掉?写在一个文件?
别团等shy哥发育
2023/02/25
7140
SpringBoot整合日志框架
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
3960
slf4j、log4j、log4j2、logback到底用哪些jar
日志级别动态调整——小工具解决大问题
背景 随着外卖业务的快速发展,业务复杂度不断增加,线上系统环境有任何细小波动,对整个外卖业务都可能产生巨大的影响,甚至形成灾难性的雪崩效应,造成巨大的经济损失。每一次客诉、系统抖动等都是对技术人员的重大考验,我们必须立即响应,快速解决问题。 如何提高排查问题的效率呢?最有效的方式是通过分析系统日志。如果系统日志全面,会为我们排查解决线上问题带来绝大的帮助,但是要想保证系统日志全面,就必须打印出所有的系统或业务日志。这样就会带来另一个问题,那就是日志量的暴涨,过多的日志除了能够帮助我们解决问题外,同时会直接造
美团技术团队
2018/03/12
2.5K1
日志级别动态调整——小工具解决大问题
Java日志框架:slf4j作用及其实现原理
slf4j是门面模式的典型应用,因此在讲slf4j前,我们先简单回顾一下门面模式,
JAVA葵花宝典
2019/05/24
1.2K0
Java日志框架学习--日志门面--中
全称为Jakarta Commons Logging,是Apache提供的一个通用日志API。
大忽悠爱学习
2022/05/16
6030
Java日志框架学习--日志门面--中
从源码来理解slf4j的绑定,以及logback对配置文件的加载
  项目中的日志系统使用的是slf4j + logback。slf4j作为一个简单日志门面,为各种loging APIs(像java.util.logging, logback, log4j)提供一个简单统一的接口,有利于维护和各个类的日志处理方式统一。Logback作为一个具体的日志组件,完成具体的日志操作。
青石路
2018/09/10
1.4K0
从源码来理解slf4j的绑定,以及logback对配置文件的加载
【编程开发】- 01 日志框架
Log4j是目前最为流行的Java日志框架之一,1999年发布首个版本,2012年发布最后一个版本,2015年正式宣布终止,官方也已不建议使用,并逐步被Logback和Log4j2等日志框架所替代,可是无法掩饰光辉历程,以及优良的设计理念。尽管Log4j有着出色的历史战绩,但早已不是Java日志框架的最优选择,还在使用该日志框架的项目往往是历史遗留问题。
Reactor2020
2023/03/22
1.3K0
【编程开发】- 01 日志框架
JAVA日志框架适配/冲突解决方案
上面的这些问题,基本都是由于多套日志框架共存或配置错误导致的。那么为什么会出现共存或者冲突呢? 一般是以下几种原因:
神秘的寇先森
2021/04/13
1.5K0
推荐阅读
相关推荐
Springboot 系列(四)Spring Boot 日志框架
更多 >
领券
社区富文本编辑器全新改版!诚邀体验~
全新交互,全新视觉,新增快捷键、悬浮工具栏、高亮块等功能并同时优化现有功能,全面提升创作效率和体验
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文