java 日志查看_Java日志-程序员宅基地

技术标签: java 日志查看  

日志对于一个系统来说非常重要,查找异常信息、分析系统运行情况等都需要用到日志。所以无论是JDK还是第三方都提供了关于日志的相关工具,本文分别介绍以下几种工具,以及各种工具间的整合、原理。

JDK的java.util.logging包

第三方日志工具(commons-logging/slf4j,log4j/logback)

JDK的java.util.logging包

JDK1.4引入了java.util.logging包,包中主要的类包括:Logger、LogManager、Handler、Formatter。首先看一段比较简单的示例代码:

packageme.likeyao.jdk.logger;importjava.util.logging.Formatter;importjava.util.logging.Handler;importjava.util.logging.Level;importjava.util.logging.LogRecord;importjava.util.logging.Logger;public classJDKLoggerTest {public static voidmain(String[] args) {

Logger logger= Logger.getLogger("logger");

logger.info("hello world");

Handler handler= newHandler() {

@Overridepublic voidpublish(LogRecord record) {

}

@Overridepublic voidflush() {

}

@Overridepublic void close() throwsSecurityException {

}

};

handler.setFormatter(newFormatter() {

@OverridepublicString format(LogRecord record) {return null;

}

});

logger.setLevel(Level.INFO);

logger.log(Level.FINEST,"hello world");

}

}

通过Logger.getLogger(name)方法可以获取logger对象,logger对象有三个比较重要的概念:level、handler、formatter。level称为日志级别,在java.util.logging包中定义了java.util.logging.Level类,里面包含SEVERE/WARNING/INFO/CONFIG/FINE/FINER/FINEST(从高到低)7种日志级别。设置日志级别会过滤掉一部分日志,例如当日志级别设置为INFO级别时,CONFIG/FINE/FINER/FINEST级别的日志就会被忽略。handler解决的问题是日志输出到哪里,是到控制台(java.util.logging.ConsoleHandler),还是到文件(java.util.logging.FileHandler),或者是写到Socket中(java.util.logging.SocketHandler)。formatter定义了日志输出的格式,可以是XML(java.util.logging.XMLFormatter),也可以自己实现JSON格式的Fomatter。

logger对象是如何生成的

生成logger对象涉及到java.util.logging.LogManager类,LogManager中用到了单例模式,在static块中初始化了LoggerManager实例对象。生成logger对象的过程:

2dd358b3b9c3c0de0a84129ff770c900.png

从图中可以看到,logger对象是在LoggerManager中创建的。LoggerManager中有一个叫userContext的LoggerContext对象,userContext缓存了所有的logger对象(缓存在namedLoggers中),并维护了一套logger对象的父子结构。namedLoggers的定义:

private final Hashtable namedLoggers = new Hashtable<>();

LoggerWeakRef继承自java.lang.ref.WeakReference,namedLoggers并不直接持有logger对象,当持有logger的对象被垃圾回收之后,只有一个weekreference指向logger,方便垃圾回收logger对象。

JDK logger其他一些有意思的东西

有一个Logger.getLogger方法是两参数的,public static Logger getLogger(String name, String resourceBundleName),第二个参数最终会变成java.util.ResourceBundle对象,可以用来做国际化。

java.util.logging.Handler可以设置java.util.logging.Filter更灵活的过滤日志。

第三方日志工具(commons-logging/slf4j,log4j/logback)

首先把四个工具分成了两组,commons-logging/slf4j和log4j/logback。log4j/logback功能与java.util.logging包类似,提供实际的日志功能。commons-logging/slf4j是门面,作用是统一日志操作,屏蔽底层不同日志组件的差异。

commons-logging

commons-logging是apache的项目,使用commons-logging的代码:

packageme.likeyao.java.logger;importorg.apache.commons.logging.Log;importorg.apache.commons.logging.LogFactory;public classJCLTest {private static Log logger = LogFactory.getLog(JCLTest.class);public static void main(String[] args) throwsException {

logger.info("hello world");

}

}

代码中用到了commons-logging日志对象(org.apache.commons.logging.Log),日志对象工厂(org.apache.commons.logging.LogFactory)。前面有说过commons-logging是一个统一操作的门面,不涉及具体的日志功能,那日志对象是怎么产生的?查看Log对象的继承关系:

0ba3d22d50e8fff723d61b1a3f0fef81.png

commons-logging分别为支持的日志工具提供了一个Log类的实现类,在列表中看到了Log4JLogger和Jdk14Logger等,意味着commons-logging可以log4j、java.util.logging组合使用。因为没有对应的logback实现,所以也就无法一起使用。下图解释commons-logging如何决定具体生成哪种Logger对象:

8c6884977f577c87d58ae0ea98510392.png

LogFactory在static块中初始化了HashTable对象factories,以ClassLoader为key,LogFactory为value缓存了所有LogFactory对象。主要看一下LogFactoryImpl的discoverLogImplementation方法是如何发现底层使用的日志工具(省略了方法一部分内容):

private static final String[] classesToDiscover ={

LOGGING_IMPL_LOG4J_LOGGER,//org.apache.commons.logging.impl.Log4JLogger

"org.apache.commons.logging.impl.Jdk14Logger","org.apache.commons.logging.impl.Jdk13LumberjackLogger","org.apache.commons.logging.impl.SimpleLog"};privateLog discoverLogImplementation(String logCategory)

...

Log result= null;//查看commons-logging.properties和System.getProperty是否配置了org.apache.commons.logging.log

String specifiedLogClassName =findUserSpecifiedLogClassName();if (specifiedLogClassName != null) {

...//如果有配置,直接使用配置的类创建Log

result =createLogFromClass(specifiedLogClassName,

logCategory,true);

...returnresult;

}

...//如果没有,遍历classesToDiscover数组,如果使用指定的ClassLoader Class.forName能加载到类,就创建Log对象

for(int i=0; i

result= createLogFromClass(classesToDiscover[i], logCategory, true);

}if (result == null) {throw newLogConfigurationException

("No suitable Log implementation");

}returnresult;

}

整个初始化Log对象的流程:

fcb2699c6f81af336458faaa37b7a84e.png

commons-logging中的classloader

网上搜commons-logging,可以看到不少文章都是在讲关于classloader的问题,贴出比较详细的一篇的链接:《Taxonomy of class loader problems encountered when using Jakarta Commons Logging》。文章中用的commons-logging本版较老,这里用commons-logging-1.2+log4k-1.2.17模拟一种场景,看看新commons-logging有什么改变。代码:

packageme.likeyao.java.logger;importjava.io.File;importjava.net.URL;importjava.net.URLClassLoader;importorg.apache.commons.logging.Log;importorg.apache.commons.logging.LogFactory;public classJCLTest3 {public static void main(String[] args) throwsException {//自定义classloader,模仿web容器classloader,自己加载优先

ChildClassLoader childClassLoader = newChildClassLoader(new URL[] { new File("c:/tmpclass/commons-logging-1.2.jar").toURL(),new File("C:/respository3/log4j/log4j/1.2.17/log4j-1.2.17.jar").toURL()});

Thread.currentThread().setContextClassLoader(childClassLoader);

Log log= LogFactory.getLog(JCLTest3.class);

log.error("hello world");

}

}class ChildClassLoader extendsURLClassLoader {publicChildClassLoader(URL[] urls) {super(urls, ClassLoader.getSystemClassLoader());

}

@Overrideprotected Class> loadClass(String name, booleanresolve)throwsClassNotFoundException {synchronized(getClassLoadingLock(name)) {

Class c=findLoadedClass(name);try{

c=findClass(name);if(resolve) {

resolveClass(c);

}

}catch(Exception e){

}if (c == null) {

c= super.loadClass(name, resolve);

}returnc;

}

}

}

命令行执行:java -cp .;commons-logging-1.2.jar me.likeyao.java.logger.JCLTest3。输出结果:执行没有抛异常,日志是通过JDK java.util.logging打出来了,而不是log4j。

c5e93e9a89e2386865dddec45a21e023.png

可以从代码中分析出为什么会产生这样的结果。这里用到了两种类加载器:系统类加载器(AppClassLoader)和自定义的ChildClassLoader,系统类加载器能加载到当前编译目录的class文件和commons-logging.jar,ChildClassLoader加载了commons-logging.jar和log4j.jar,线程上下文加载器被设置为ChildClassLoader。当运行java命令时,随着程序运行会使用系统类加载器加载Log类、LogFactory类,查看上面discoverLogImplementation方法源码,由于没有配置org.apache.commons.logging.log属性,具体使用哪个Log的实现类会通过遍历classesToDiscover数组决定。下面看一下classesToDiscover[0]时的情况:

private Log createLogFromClass(String logAdapterClassName, //org.apache.commons.logging.impl.Log4JLogger

String logCategory,booleanaffectState)throwsLogConfigurationException {

...//通过logAdapterClassName所制定的类创建的对象,方法返回的结果

Log logAdapter = null;

...//获取classloader,这里会返回ChildClassLoader,即main方法中设置的线程上下文加载器

ClassLoader currentCL =getBaseClassLoader();for(;;) {try{

...

Class c;try{//因为ChildClassLoader加载器可以获取到org.apache.commons.logging.impl.Log4JLogger类,正常得到c

c = Class.forName(logAdapterClassName, true, currentCL);

}catch(ClassNotFoundException originalClassNotFoundException) {

...

}//创建org.apache.commons.logging.impl.Log4JLogger对象,这里o的classloader是ChildClassLoader

constructor =c.getConstructor(logConstructorSignature);

Object o=constructor.newInstance(params);//注意下面的注释,因为Log和o的类加载器不一致,所以不进入if分支//Note that we do this test after trying to create an instance//[rather than testing Log.class.isAssignableFrom(c)] so that//we don't complain about Log hierarchy problems when the//adapter couldn't be instantiated anyway.

if (o instanceofLog) {

logAdapterClass=c;

logAdapter=(Log) o;break;

}//Oops, we have a potential problem here. An adapter class//has been found and its underlying lib is present too, but//there are multiple Log interface classes available making it//impossible to cast to the type the caller wanted. We//certainly can't use this logger, but we need to know whether//to keep on discovering or terminate now.//

//The handleFlawedHierarchy method will throw//LogConfigurationException if it regards this problem as//fatal, and just return if not.//根据allowFlawedHierarchy参数判断是否抛出LogConfigurationException,如果不抛异常,只是简单return,进入下一个循环

handleFlawedHierarchy(currentCL, c);

}catch(NoClassDefFoundError e) {

...break;

}catch(ExceptionInInitializerError e) {

...break;

}catch(LogConfigurationException e) {//call to handleFlawedHierarchy above must have thrown//a LogConfigurationException, so just throw it on

throwe;

}catch(Throwable t) {

...

}if (currentCL == null) {break;

}//try the parent classloader//currentCL = currentCL.getParent();

currentCL =getParentClassLoader(currentCL);

}

...returnlogAdapter;

}

classesToDiscover[0]的调用中,logAdapter并没有指向org.apache.commons.logging.impl.Log4JLogger对象,而是返回了null。所以进入下一次循环classesToDiscover[1](org.apache.commons.logging.impl.Jdk14Logger),这一次因为Log类和org.apache.commons.logging.impl.Jdk14Logger类都是由系统类加载器加载,所以最终执行的结果日志由java.util.logging打出。commons-logging使用classloader来加载Resource和发现底层具体日志工具,在web容器或者OSGI这些需要用类加载器做隔离的情况下的确会出现一些问题。

SLF4J/LOG4J

另一种日志门面SLF4J,它采用“静态绑定”的方式避免了commons-logging中有关类加载的一些问题。下面的把SLF4J和LOG4J放到一起,首先还是比较简单的使用代码:

packageme.likeyao.slf4j.logger;importorg.slf4j.Logger;importorg.slf4j.LoggerFactory;public classSLF4JTest {public static voidmain(String[] args) {

Logger logger= LoggerFactory.getLogger(SLF4JTest.class);

logger.error("hello world");

}

}

使用方式上看与commons-logging类似,除了类名上有一些区别。下图的是SLF4J整合LOG4J时Logger初始化时序图:

9507b6680b71411f8485ff0005b691b6.png

这里对几个类用了不同的颜色,蓝色代表类在slf4j-api包中,黄色代表类在slf4j-log4j12包中,红色代表类在log4j包中。相比commons-logging运行时通过classloader来发现底层日志工具,SLF4J是通过在不同的“桥梁包”里放置同名类org.slf4j.impl.StaticLoggerBinder来实现日志工具的绑定。例如,如果需要将SLF4J与java.util.logging整合,需要加入slf4j-jdk14-1.7.12,包中的StaticLoggerBinder类返回的loggerfactory是org.slf4j.impl.JDK14LoggerFactory;如果是SLF4J与log4j整合,需要加入slf4j-log4j12-1.7.12,包中的StaticLoggerBinder类返回的loggerfactory是org.slf4j.impl.Log4jLoggerFactory。这就是SLF4J的静态绑定。

LOG4J是常用的日志工具,与java.util.logging(jul)非常像,一些概念也是共通的。例如logger都是父子结构的、jul的handler对应log4j的appender、formater对应layout、也都存在filter。为了理清log4j中各个类的关系,整理一份类图:

bd9488fb6a377d967cbe0233e4586eaa.png

整个类图的核心是org.apache.log4j.Category,但从1.2版本后log4j不会直接产生Category对象,而是Logger对象。Logger类继承自Category,扩展了一个日志级别:trace。当调用Logger对象的info/warn等方法时会生成一个LoggerEvent对象,AppenderAttachableImpl会遍历所有的appender调用doAppend方法。如果event没有被Filter过滤掉,那最终会经过Layout格式化,输出到appender指定的地方。LOG4J提供了很多appender供使用,这一点比JDK的hander强大很多。

Appender的初始化在LogManager的static块中进行,最终解析发生在org.apache.log4j.xml.DOMConfigurator类中,logger、root、appender都在这里解析。

if (tagName.equals(CATEGORY) ||tagName.equals(LOGGER)) {

parseCategory(currentElement);

}else if(tagName.equals(ROOT_TAG)) {

parseRoot(currentElement);

}else if(tagName.equals(RENDERER_TAG)) {

parseRenderer(currentElement);

}else if(tagName.equals(THROWABLE_RENDERER_TAG)) {if (repository instanceofThrowableRendererSupport) {

ThrowableRenderer tr=parseThrowableRenderer(currentElement);if (tr != null) {

((ThrowableRendererSupport) repository).setThrowableRenderer(tr);

}

}

}else if (!(tagName.equals(APPENDER_TAG)||tagName.equals(CATEGORY_FACTORY_TAG)||tagName.equals(LOGGER_FACTORY_TAG))) {

quietParseUnrecognizedElement(repository, currentElement, props);

}

LOGBACK

logback是log4j的一个替代品,初始化和打日志代码流程相似。为什么要从log4j切换到logback,logback网站上已经给出了(Reasons to prefer logback over log4j)。要使用SLF4J+logback需要引入三个包:slf4j-api、logback-core、logback-classic。这里和log4j对比,介绍一下两种日志工具的父子结构。无论是java.util.logging还是log4j/logback都为logger对象提供了父子结构,这样做有什么好处?我觉得主要是这样logger对象会有一个树形的层次结构,底层的logger可以复用父logger中的一些配置,比如日志级别,appender等。从log4j和logback的配置文件中,也可以看出这一点,通过为某个包名的logger指定appender和日志级别,可以作用所有这个包下的logger。例如:

log4j和logback在处理父子结构时有一些差别,先看log4j的代码:

synchronized(ht) {

Object o=ht.get(key);if(o == null) {

logger=factory.makeNewLoggerInstance(name);

logger.setHierarchy(this);

ht.put(key, logger);

updateParents(logger);returnlogger;

}else if(o instanceofLogger) {return(Logger) o;

}else if (o instanceofProvisionNode) {//System.out.println("("+name+") ht.get(this) returned ProvisionNode");

logger =factory.makeNewLoggerInstance(name);

logger.setHierarchy(this);

ht.put(key, logger);

updateChildren((ProvisionNode) o, logger);

updateParents(logger);returnlogger;

}else{//It should be impossible to arrive here

return null; //but let's keep the compiler happy.

}

}

log4j在创建logger时,如果父节点是一个logger,那只维护一个子logger对父logger的引用。如果父节点不存在,就创建一些ProvisionNode对象(这是一个Vector的子类),保存所有下面子的logger。举个例子:当获取名称为x.y.z的logger时,首先会创建logger(x.y.z);然后查找父logger x.y,如果x.y不存在,就创建一个ProvisionNode对象,把logger(x.y.z)放到ProvisionNode对象中;然后继续向上搜索名称为x的logger,就将logger(x.y.z)的parent设置为logger(x)。当调用logger(x.y.z)对象的info/warn等方法时,如果logger(x.y.z)没有设置日志级别和appender,就是沿着parent向上搜索,直到rootLogger为止。

logback的结构稍微有一些区别,所有的节点都是Logger对象,对象中有指向parent和children的引用,并且创建节点时,会把parent的日志级别直接复制到自己对象中。

//if the desired logger does not exist, them create all the loggers//in between as well (if they don't already exist)

String childName;while (true) {int h =LoggerNameUtil.getSeparatorIndexOf(name, i);if (h == -1) {

childName=name;

}else{

childName= name.substring(0, h);

}//move i left of the last point

i = h + 1;synchronized(logger) {

childLogger=logger.getChildByName(childName);if (childLogger == null) {

childLogger=logger.createChildByName(childName);

loggerCache.put(childName, childLogger);

incSize();

}

}

logger=childLogger;if (h == -1) {returnchildLogger;

}

}

总结

SL4J/COMMONS-LOGGING、LOG4J、LOGBACK、JUL都是常用的Java日志工具,基本思想都是通过Factory生成logger对象,然后由LogManager管理/缓存logger对象,同时维护一个父子结构方便复用配置。logger对象包含三要素:日志级别、输出到哪里、格式化。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_30082367/article/details/114060708

智能推荐

从零开始搭建Hadoop_创建一个hadoop项目-程序员宅基地

文章浏览阅读331次。第一部分:准备工作1 安装虚拟机2 安装centos73 安装JDK以上三步是准备工作,至此已经完成一台已安装JDK的主机第二部分:准备3台虚拟机以下所有工作最好都在root权限下操作1 克隆上面已经有一台虚拟机了,现在对master进行克隆,克隆出另外2台子机;1.1 进行克隆21.2 下一步1.3 下一步1.4 下一步1.5 根据子机需要,命名和安装路径1.6 ..._创建一个hadoop项目

心脏滴血漏洞HeartBleed CVE-2014-0160深入代码层面的分析_heartbleed代码分析-程序员宅基地

文章浏览阅读1.7k次。心脏滴血漏洞HeartBleed CVE-2014-0160 是由heartbeat功能引入的,本文从深入码层面的分析该漏洞产生的原因_heartbleed代码分析

java读取ofd文档内容_ofd电子文档内容分析工具(分析文档、签章和证书)-程序员宅基地

文章浏览阅读1.4k次。前言ofd是国家文档标准,其对标的文档格式是pdf。ofd文档是容器格式文件,ofd其实就是压缩包。将ofd文件后缀改为.zip,解压后可看到文件包含的内容。ofd文件分析工具下载:点我下载。ofd文件解压后,可以看到如下内容: 对于xml文件,可以用文本工具查看。但是对于印章文件(Seal.esl)、签名文件(SignedValue.dat)就无法查看其内容了。本人开发一款ofd内容查看器,..._signedvalue.dat

基于FPGA的数据采集系统(一)_基于fpga的信息采集-程序员宅基地

文章浏览阅读1.8w次,点赞29次,收藏313次。整体系统设计本设计主要是对ADC和DAC的使用,主要实现功能流程为:首先通过串口向FPGA发送控制信号,控制DAC芯片tlv5618进行DA装换,转换的数据存在ROM中,转换开始时读取ROM中数据进行读取转换。其次用按键控制adc128s052进行模数转换100次,模数转换数据存储到FIFO中,再从FIFO中读取数据通过串口输出显示在pc上。其整体系统框图如下:图1:FPGA数据采集系统框图从图中可以看出,该系统主要包括9个模块:串口接收模块、按键消抖模块、按键控制模块、ROM模块、D.._基于fpga的信息采集

微服务 spring cloud zuul com.netflix.zuul.exception.ZuulException GENERAL-程序员宅基地

文章浏览阅读2.5w次。1.背景错误信息:-- [http-nio-9904-exec-5] o.s.c.n.z.filters.post.SendErrorFilter : Error during filteringcom.netflix.zuul.exception.ZuulException: Forwarding error at org.springframework.cloud..._com.netflix.zuul.exception.zuulexception

邻接矩阵-建立图-程序员宅基地

文章浏览阅读358次。1.介绍图的相关概念  图是由顶点的有穷非空集和一个描述顶点之间关系-边(或者弧)的集合组成。通常,图中的数据元素被称为顶点,顶点间的关系用边表示,图通常用字母G表示,图的顶点通常用字母V表示,所以图可以定义为:  G=(V,E)其中,V(G)是图中顶点的有穷非空集合,E(G)是V(G)中顶点的边的有穷集合1.1 无向图:图中任意两个顶点构成的边是没有方向的1.2 有向图:图中..._给定一个邻接矩阵未必能够造出一个图

随便推点

MDT2012部署系列之11 WDS安装与配置-程序员宅基地

文章浏览阅读321次。(十二)、WDS服务器安装通过前面的测试我们会发现,每次安装的时候需要加域光盘映像,这是一个比较麻烦的事情,试想一个上万个的公司,你天天带着一个光盘与光驱去给别人装系统,这将是一个多么痛苦的事情啊,有什么方法可以解决这个问题了?答案是肯定的,下面我们就来简单说一下。WDS服务器,它是Windows自带的一个免费的基于系统本身角色的一个功能,它主要提供一种简单、安全的通过网络快速、远程将Window..._doc server2012上通过wds+mdt无人值守部署win11系统.doc

python--xlrd/xlwt/xlutils_xlutils模块可以读xlsx吗-程序员宅基地

文章浏览阅读219次。python–xlrd/xlwt/xlutilsxlrd只能读取,不能改,支持 xlsx和xls 格式xlwt只能改,不能读xlwt只能保存为.xls格式xlutils能将xlrd.Book转为xlwt.Workbook,从而得以在现有xls的基础上修改数据,并创建一个新的xls,实现修改xlrd打开文件import xlrdexcel=xlrd.open_workbook('E:/test.xlsx') 返回值为xlrd.book.Book对象,不能修改获取sheett_xlutils模块可以读xlsx吗

关于新版本selenium定位元素报错:‘WebDriver‘ object has no attribute ‘find_element_by_id‘等问题_unresolved attribute reference 'find_element_by_id-程序员宅基地

文章浏览阅读8.2w次,点赞267次,收藏656次。运行Selenium出现'WebDriver' object has no attribute 'find_element_by_id'或AttributeError: 'WebDriver' object has no attribute 'find_element_by_xpath'等定位元素代码错误,是因为selenium更新到了新的版本,以前的一些语法经过改动。..............._unresolved attribute reference 'find_element_by_id' for class 'webdriver

DOM对象转换成jQuery对象转换与子页面获取父页面DOM对象-程序员宅基地

文章浏览阅读198次。一:模态窗口//父页面JSwindow.showModalDialog(ifrmehref, window, 'dialogWidth:550px;dialogHeight:150px;help:no;resizable:no;status:no');//子页面获取父页面DOM对象//window.showModalDialog的DOM对象var v=parentWin..._jquery获取父window下的dom对象

什么是算法?-程序员宅基地

文章浏览阅读1.7w次,点赞15次,收藏129次。算法(algorithm)是解决一系列问题的清晰指令,也就是,能对一定规范的输入,在有限的时间内获得所要求的输出。 简单来说,算法就是解决一个问题的具体方法和步骤。算法是程序的灵 魂。二、算法的特征1.可行性 算法中执行的任何计算步骤都可以分解为基本可执行的操作步,即每个计算步都可以在有限时间里完成(也称之为有效性) 算法的每一步都要有确切的意义,不能有二义性。例如“增加x的值”,并没有说增加多少,计算机就无法执行明确的运算。 _算法

【网络安全】网络安全的标准和规范_网络安全标准规范-程序员宅基地

文章浏览阅读1.5k次,点赞18次,收藏26次。网络安全的标准和规范是网络安全领域的重要组成部分。它们为网络安全提供了技术依据,规定了网络安全的技术要求和操作方式,帮助我们构建安全的网络环境。下面,我们将详细介绍一些主要的网络安全标准和规范,以及它们在实际操作中的应用。_网络安全标准规范