log4j源码解析-文件解析
承接前文log4j源码解析,前文主要介绍了log4j的文件加载方式以及Logger对象创建。本文将在此基础上具体看下log4j是如何解析文件并输出我们所常见的日志格式
附例
文件的加载方式,我们就选举log4j.properties
作为分析的文件例子,并附上相应的通用配置
log4j.rootLogger=info,stdout,logfile,errorfile
log4j.logger.org.apache=DEBUG
log4j.logger.java.sql.Connection=DEBUG
log4j.logger.java.sql.Statement=DEBUG
log4j.logger.java.sql.PreparedStatement=DEBUG
log4j.logger.java.sql.ResultSet=INFO
log4j.logger.freemarker.core=error
#standout log appender #
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%d %p [%c] - %m%n
#common log appender #
log4j.appender.logfile=org.apache.log4j.DailyRollingFileAppender
log4j.appender.logfile.File=../logs/appender-test/info.log
log4j.appender.logfile.append=true
log4j.appender.logfile.encoding=GB18030
log4j.appender.logfile.layout=org.apache.log4j.PatternLayout
log4j.appender.logfile.layout.ConversionPattern=%d %p [%c] - %m%n
#error log appender #
log4j.appender.errorfile=org.apache.log4j.DailyRollingFileAppender
log4j.appender.errorfile.File=../logs/appender-test/error.log
log4j.appender.errorfile.Threshold=WARN
log4j.appender.errorfile.append=true
log4j.appender.errorfile.encoding=GB18030
log4j.appender.errorfile.layout=org.apache.log4j.PatternLayout
log4j.appender.errorfile.layout.ConversionPattern=%d %p [%c] - %m%n
此处不详解,我们直接看源码方面是如何处理,从代码层面来通用理解下上述的配置
OptionConverter
操作log4j配置的主要工具类,所有的读取配置并封装成对象均以此类作为入口,其在LogManager的调用方式为
OptionConverter.selectAndConfigure(url,configuratorClassName,LogManager.getLoggerRepository());
入参作下简单的展示
url - 文件路径
clazz - log4j.configuratorClass属性对应的class,默认为null
hierarchy - log4j的层级管理类,存储log4j的通用配置,默认为Hierarchy类
OptionConverter#selectAndConfigure-解析配置入口
简单看下源码
static public void selectAndConfigure(URL url, String clazz, LoggerRepository hierarchy) {
Configurator configurator = null;
String filename = url.getFile();
//xml格式的文件则采用DOMConfigurator解析类,表明默认采用xml格式的解析方式
if(clazz == null && filename != null && filename.endsWith(".xml")) {
clazz = "org.apache.log4j.xml.DOMConfigurator";
}
if(clazz != null) {
LogLog.debug("Preferred configurator class: " + clazz);
configurator = (Configurator) instantiateByClassName(clazz,
Configurator.class,
null);
if(configurator == null) {
LogLog.error("Could not instantiate configurator ["+clazz+"].");
return;
}
} else {
//最后一种方式则为properties解析方式
configurator = new PropertyConfigurator();
}
configurator.doConfigure(url, hierarchy);
}
从简单的注释中我们可以得出log4j
只支持两种方式的解析方式
1.DOMConfigurator-xml格式的解析器,默认
2.PropertyConfigurator-properties格式的解析器
本文则着重讲解.properties
配置文件的解析,即关注PropertyConfigurator
解析器
PropertyConfigurator#doConfigure-解析properties配置文件
读取文件的方式就不分析了,很常见的采用Properties
类来存储数据,递上重要的逻辑片段代码
public void doConfigure(Properties properties, LoggerRepository hierarchy) {
repository = hierarchy;
// 读取log4j.debug配置,值为boolean型,表明内部log是否支持debug模式
String value = properties.getProperty(LogLog.DEBUG_KEY);
if(value == null) {
value = properties.getProperty("log4j.configDebug");
if(value != null)
LogLog.warn("[log4j.configDebug] is deprecated. Use [log4j.debug] instead.");
}
if(value != null) {
LogLog.setInternalDebugging(OptionConverter.toBoolean(value, true));
}
//读取log4j.reset的boolean值,true代表使用默认的配置
String reset = properties.getProperty(RESET_KEY);
if (reset != null && OptionConverter.toBoolean(reset, false)) {
hierarchy.resetConfiguration();
}
//log4j.threshold阈值配置,也就是告警级别配置
String thresholdStr = OptionConverter.findAndSubst(THRESHOLD_PREFIX,
properties);
if(thresholdStr != null) {
hierarchy.setThreshold(OptionConverter.toLevel(thresholdStr,
(Level) Level.ALL));
LogLog.debug("Hierarchy threshold set to ["+hierarchy.getThreshold()+"].");
}
// 配置根分类,也就是rootLogger
configureRootCategory(properties, hierarchy);
// 配置Logger工厂
configureLoggerFactory(properties);
// 解析非root的其他配置
parseCatsAndRenderers(properties, hierarchy);
LogLog.debug("Finished configuring.");
// 清空下缓存
registry.clear();
}
按照上面的解析顺序作下备注
1.解析
log4j.debug/log4j.configDebug(boolean)
是否让log4j的内部输出源支持debug模式,其实也就是是否调用System.out.println()方法,默认不支持debug模式,支持warn/error模式。(支持System.err.println()方法)
2.
解析log4j.reset(boolean)
是否重新设置log4j配置,默认不重新设置(Optional,作用微小)
3.
解析log4j.threshold(String)
trace/debug/info/warn/error/fatal配置告警级别,表明对所有的输出源,低于该等级则不输出
4.解析根节点rootLogger
5.解析日志工厂
6.解析非根节点
我们对后三步的操作作下简单的分析,加深我们对通用配置的理解
PropertyConfigurator#configureRootCategory-解析根节点
首先简单的看下里面的操作逻辑
void configureRootCategory(Properties props, LoggerRepository hierarchy) {
// log4j.rootLogger或者log4j.rootCategory,支持${}系统变量取值
String effectiveFrefix = ROOT_LOGGER_PREFIX;
String value = OptionConverter.findAndSubst(ROOT_LOGGER_PREFIX, props);
if(value == null) {
value = OptionConverter.findAndSubst(ROOT_CATEGORY_PREFIX, props);
effectiveFrefix = ROOT_CATEGORY_PREFIX;
}
if(value == null)
LogLog.debug("Could not find root logger information. Is this OK?");
else {
Logger root = hierarchy.getRootLogger();
synchronized(root) {
// 关键代码
parseCategory(props, root, effectiveFrefix, INTERNAL_ROOT_NAME, value);
}
}
}
承接上述的关键代码分析,此处的logger
参数为rootLogger
/**
** This method must work for the root category as well.
*/
void parseCategory(Properties props, Logger logger, String optionKey,
String loggerName, String value) {
LogLog.debug("Parsing for [" +loggerName +"] with value=[" + value+"].");
// ,分隔符解析
StringTokenizer st = new StringTokenizer(value, ",");
if(!(value.startsWith(",") || value.equals(""))) {
if(!st.hasMoreTokens())
return;
String levelStr = st.nextToken();
LogLog.debug("Level token is [" + levelStr + "].");
// If the level value is inherited, set category level value to
// null. We also check that the user has not specified inherited for the
// root category.
if(INHERITED.equalsIgnoreCase(levelStr) ||
NULL.equalsIgnoreCase(levelStr)) {
if(loggerName.equals(INTERNAL_ROOT_NAME)) {
LogLog.warn("The root logger cannot be set to null.");
} else {
logger.setLevel(null);
}
} else {
logger.setLevel(OptionConverter.toLevel(levelStr, (Level) Level.DEBUG));
}
LogLog.debug("Category " + loggerName + " set to " + logger.getLevel());
}
// 删除所有的输出源对象
logger.removeAllAppenders();
Appender appender;
String appenderName;
while(st.hasMoreTokens()) {
appenderName = st.nextToken().trim();
if(appenderName == null || appenderName.equals(","))
continue;
LogLog.debug("Parsing appender named \"" + appenderName +"\".");
appender = parseAppender(props, appenderName);
if(appender != null) {
logger.addAppender(appender);
}
}
}
由以上的代码可以简单的得知log4j.rootLogger对应的配置项为{level},{appenderNames}
1.{level} - 日志等级,设置根日志的日志等级,应用于所有的输出源
2.{appenderNames} - 可配置多个输出源,以
,
为分隔符。并由此属性解析log4j.appender开头的配置项
再而分析了解下PropertyConfigurator#parseAppender()
方法解析输出源,为了防止代码展示过多,我们截取主要的代码片段进行分析
Appender parseAppender(Properties props, String appenderName) {
....
// log4j.appender.{appenderName}
String prefix = APPENDER_PREFIX + appenderName;
// log4j.appender.{appenderName}.layout
String layoutPrefix = prefix + ".layout";
// 首先根据log4j.appender.{appenderName}解析得到Appender对象
appender = (Appender) OptionConverter.instantiateByKey(props, prefix,
org.apache.log4j.Appender.class,
null);
...
if(appender instanceof OptionHandler) {
if(appender.requiresLayout()) {
// 解析得到Layout对象,代表该输出源的输出格式
Layout layout = (Layout) OptionConverter.instantiateByKey(props,
layoutPrefix,
Layout.class,
null);
....
// 解析log4j.appender.{appenderName}.errorhandler
final String errorHandlerPrefix = prefix + ".errorhandler";
String errorHandlerClass = OptionConverter.findAndSubst(errorHandlerPrefix, props);
if (errorHandlerClass != null) {
ErrorHandler eh = (ErrorHandler) OptionConverter.instantiateByKey(props,
errorHandlerPrefix,
ErrorHandler.class,
null);
if (eh != null) {
appender.setErrorHandler(eh);
LogLog.debug("Parsing errorhandler options for \"" + appenderName +"\".");
// 解析ErrorHandler对象
parseErrorHandler(eh, errorHandlerPrefix, props, repository);
...
}
}
...
}
// 解析log4j.appender.{appenderName}.filter配置
parseAppenderFilters(props, appenderName, appender);
registryPut(appender);
return appender;
}
具体解析的过程就不讲解了,只在此处作下罗列,此处假定appenderName为
console
1.log4j.appender.console - 对应的输出源的类名
2.log4j.appender.console.layout - 对应输出源的日志展示类名,通用为
log4j.appender.{appenderName}.layout
3.log4j.appender.console.errorhandler-对应输出源的错误信息处理类名
4.log4j.appender.console.filter-输出源过滤类,支持配置多个。格式为log4j.appender.console.filter.{filterName}={filterClass}
5.log4j.appender.console.encoding/threshold-此类的额外参数,其会利用反射的机制调用相应的setter方法进行设置
PropertyConfigurator#configureLoggerFactory-解析日志工厂
解析的为log4j.loggerFactory配置,其可以指定logger工厂的实现类,默认为
DefaultCategoryFactory
,其内部就一个方法makeNewLoggerInstance()
用于创建日志类Logger。用户可自定义实现
PropertyConfigurator#parseCatsAndRenderers-解析非根节点
解析的为log4j.logger/log4j.category/log4j.additivity/log4j.renderer/log4j.throwableRenderer
配置,具体解析读者可自行分析,此处作下总结
1.
log4j.logger/log4j.category
以此为开头的配置,其会解析为Logger对象,方式与log4j.rootLogger
配置一致,多用于对指定的类进行特定级别的输出,默认继承根Logger对象的输出源配置2.
log4j.additivity
以此开头的配置,表明对特定的Logger对象只输出自己所拥有的Appenders,不采用根Logger对象的Appenders3.
log4j.renderer/log4j.throwableRenderer
开头的配置,前者主要配置对普通输出信息的渲染处理,后者对异常信息的渲染处理。默认均有实现,一般不指定4.对输出源的日志级别输出与否比较规则作下总结(以基于
com.jing.test.Application
类调用info()方法输出举个例子):
首先判断是否>=
log4j.threshold
属性指定的日志级别Level,如果不满足,则不输出,反之继续往下走。eg.log4j.threshold=WARN
则输出源无法输出然后根据loggerName获取
log4j.category
/log4j.logger
对应的Level(如果loggerName以.分隔,则当前loggerName找不到会向父级获取,没定义则应用rootLogger),判断是否>=Level,否则不输出,反之继续往下走
eg. 比如定义了log4j.logger.com.jing.test=INFO
,但没定义log4j.logger.com.jing.test.Application
则其会应用com.jing.test
中的Level,即Level=INFO最后获取输出源Appender指定的日志级别Level,即
${appenderName}.Threshold
属性,如果>=指定的Level,则进行输出,反之不输出
总结
见本文的分析,通过源码加深我们对配置的理解,心中多一份踏实
log4j源码解析-文件解析的更多相关文章
- MyBatis 源码分析 - 配置文件解析过程
* 本文速览 由于本篇文章篇幅比较大,所以这里拿出一节对本文进行快速概括.本篇文章对 MyBatis 配置文件中常用配置的解析过程进行了较为详细的介绍和分析,包括但不限于settings,typeAl ...
- InfluxDB源码目录结构解析
操作系统 : CentOS7.3.1611_x64 go语言版本:1.8.3 linux/amd64 InfluxDB版本:1.1.0 influxdata主目录结构 [root@localhost ...
- Spring源码入门——DefaultBeanNameGenerator解析 转发 https://www.cnblogs.com/jason0529/p/5272265.html
Spring源码入门——DefaultBeanNameGenerator解析 我们知道在spring中每个bean都要有一个id或者name标示每个唯一的bean,在xml中定义一个bean可以指 ...
- 鸿蒙内核源码分析(ELF解析篇) | 你要忘了她姐俩你就不是银 | 百篇博客分析OpenHarmony源码 | v53.02
百篇博客系列篇.本篇为: v53.xx 鸿蒙内核源码分析(ELF解析篇) | 你要忘了她姐俩你就不是银 | 51.c.h.o 加载运行相关篇为: v51.xx 鸿蒙内核源码分析(ELF格式篇) | 应 ...
- v72.01 鸿蒙内核源码分析(Shell解析) | 应用窥伺内核的窗口 | 百篇博客分析OpenHarmony源码
子曰:"苟正其身矣,于从政乎何有?不能正其身,如正人何?" <论语>:子路篇 百篇博客系列篇.本篇为: v72.xx 鸿蒙内核源码分析(Shell解析篇) | 应用窥视 ...
- 笔记-twisted源码-import reactor解析
笔记-twisted源码-import reactor解析 1. twisted源码解析-1 twisted reactor实现原理: 第一步: from twisted.internet ...
- java 日志体系(四)log4j 源码分析
java 日志体系(四)log4j 源码分析 logback.log4j2.jul 都是在 log4j 的基础上扩展的,其实现的逻辑都差不多,下面以 log4j 为例剖析一下日志框架的基本组件. 一. ...
- [C#] .NET Core项目修改project.json来引用其他目录下的源码等文件的办法 & 解决多框架时 project.json 与 app.config冲突的问题
作者: zyl910 一.缘由 项目规模大了后,经常会出现源码文件分布在不同目录的情况,但.NET Core项目默认只有项目目录下的源码文件,且不支持"Add As Link"方式 ...
- .NET Core项目修改project.json来引用其他目录下的源码等文件的办法 & 解决多框架时 project.json 与 app.config冲突的问题
作者: zyl910 一.缘由 项目规模大了后,经常会出现源码文件分布在不同目录的情况,但.NET Core项目默认只有项目目录下的源码文件,且不支持“Add As Link”方式引入文件.这时需要手 ...
随机推荐
- Shell菜单脚本
今天在这儿给大家分享一个我简单编写的Shell菜单脚本,傻瓜式的人机交互,人人都可以操作linux. #!/bin/sh #Shell菜单演示 function menu () { cat <& ...
- python面向对象进阶
前言 上节大话python面向对象对面向对象有了一些了解,这次就不用大话风格了 (ps:真心不好扯啊) isinstance与issubclass isinstance(obj,cls)检查是否obj ...
- 五 : springMVC拦截器
springMVC拦截器的实现一般有两种方式 第一种方式是要定义的Interceptor类要实现了Spring的HandlerInterceptor 接口 第二种方式是继承实现了HandlerInte ...
- C#面试题整理(1)
最近在看CLR VIA C#,发现了一些案例很适合来做面试题.特此整理: 1,System.Object里的GetType方法是否为虚函数?说出理由. 答案:不是,因为C#是一种类型安全的语言,如果覆 ...
- 浅谈event.client、event.screen与event.offset
每每看到event.client.event.screen与event.offset这几个,头都大了,今天又碰到了,特来总结下. 1.event.screenX与event.screenY. 首先,e ...
- java通过smtp发送电子邮件
package com.sm.modules.oa.web; import javax.mail.Session; import javax.mail.Transport; import javax. ...
- XML中文乱码问题
XML中文乱码问题
- Sublime Text 使用介绍、全套快捷键及插件推荐
开篇:如果说Notepad++是一款不错Code神器,那么Sublime Text应当称得上是神器滴哥.Sublime Text最大的优点就是跨平台,Mac和Windows均可完美使用:其次是强大的插 ...
- Lytro 光场相机重对焦C++实现以及CUDA实现
前面有几篇博客主要介绍了光场和光场相机相关知识,以及重对焦效果和多视角效果的展示.算是自己学习光场过程的一种总结. 这次贴上自己用OpenCV/C++编写的重对焦算法实现(包含CPU版和CUDA GP ...
- Java多线程异常处理
在java多线程程序中,所有线程都不允许抛出未捕获的checked exception,也就是说各个线程需要自己把自己的checked exception处理掉.这一点是通过java.lang.Run ...