Log4net源码View之Logger解析

摘要:
1.简介Logger是Log4net的三大核心载体之一。理解它具有重要意义。另外两个是Appender和Layout。对应的关系是一个Logger对应多个Appender,一个Appenders对应一个Layout。对于Log4net的内部实现,ILogger是一个接口(所有日志都需要实现——Interfacethatloggersimple)。为什么不是我们使用的ILog接口

 

1.简介

Logger是Log4net的三大核心载体之一,搞清楚它的意义很重大。另外两个分别是AppenderLayout。其对应关系为一个Logger对应多个Appender,一个Appender对应一个Layout。

对于Log4net的内部实现,ILogger才是接口(所有日志需要实现的——Interface that all loggers implement)。

为什么不是我们所使用的ILog接口呢?实际上,ILog是对ILogger进行了包装,是典型的Wrapper模式,话说不仔细看的话,还真搞不清楚它们的关系。

下图表明了关系:ILog——>ILoggerWrapper——>ILogger.

Log4net源码View之Logger解析第1张

Log4net源码View之Logger解析第2张

所以,从下面开始,请先忽略ILog接口,将关注点放在ILogger上,这也是本文的重点。

2.ILogger接口

方法

含义

string Name{ get; }

获取ILogger的名称

Log(Type callerStackBoundaryDeclaringType, Level level, object message, Exception exception);

将需要记录的信息(level、msg、ex)变成一个LoggerEvent对象,然后将LoggerEvent对象分发到各个Appender。

void Log(LoggingEvent logEvent);

直接记录一个LoggerEvent

bool IsEnabledFor(Level level);

检查某级别是否可用

ILoggerRepository Repository { get; }

ILogger所在的仓储

从接口结合源码可以得出:

2.1 仓储

通过Repository可以找到ILogger所在的仓储,得到对应的分组(Repository Group);

设计思想

将多个日志对象进行分组,以便于管理。比如用户User,用户组UserGroup,其中UserGroup就是分组。

而Repository除了有分组的意义之外,还有存储的作用,可以理解为带有缓存功能的工厂(Factory+Cache)。

 

2.2 LoggerEvent

LoggerEvent又是什么呢?(很早以前看到LoggerEvent时,一直很纠结为什么要这样取名?是为了让别人看不懂源码吗?仔细揣摩之后,比如WCF的Messaging,EBaySDK中的Request,只是某个语境下的一个特性称谓)

设计思想

LoggerEvent可以理解为一个信封Envelope,类似于WebService中soap的概念,也类似于MessageQueue中的Messaging的设计。其主要用来包装与消息(object message)有关的数据。这样,通过将LoggerEvent分发给Appender,就实现了输出到多个终端。如下代码:

  1. virtualprotectedvoid CallAppenders(LoggingEvent loggingEvent)
  2. {
  3. //......
  4.     int writes = 0;
  5.     for (Logger c = this; c != null; c = c.Parent)
  6.     {
  7.         if (c.m_appenderAttachedImpl != null)
  8.         {
  9.             // Protected against simultaneous call to addAppender, removeAppender,...
  10.             c.m_appenderLock.AcquireReaderLock();
  11.             try
  12.             {
  13.                 if (c.m_appenderAttachedImpl != null)
  14.                 {
  15.                     writes += c.m_appenderAttachedImpl.AppendLoopOnAppenders(loggingEvent);
  16.                 }
  17.             }
  18.             finally
  19.             {
  20.                 c.m_appenderLock.ReleaseReaderLock();
  21.             }
  22.         }
  23.        //......
  24.     }
  25. }

其中,m_appenderAttachedImpl为Appender的集合,AppendLoopOnAppenders循环将LoggerEvent添加到Appender。

 

2.3 LoggerEventData

"信封"里的数据是什么呢?——LoggerEventData对象。

  1. publicstruct LoggingEventData
  2. {
  3.    publicstring LoggerName;
  4.    public Level Level;
  5.    publicstringMessage;
  6.    publicstring ThreadName;
  7.    public DateTime TimeStamp;
  8.    public LocationInfo LocationInfo;
  9.    publicstring UserName;
  10.    publicstring Identity;
  11.    publicstringExceptionString;
  12.    publicstring Domain;
  13.    public PropertiesDictionary Properties;
  14. }

最常用的数据,TimeStamp,Message,ExceptionString

3.抽象类Logger

Log4net源码View之Logger解析第3张

特点:

1)通过继承IAppenderAttachable接口,这样Logger与IAppender之间就有一对多的包含与调用关系;

2)Logger具有层次结构,公布了类型为Logger的父亲指针Parent;

3)通过实现ILogger(具体是使用了LoggerEvent对象),则将日志的整体包装起来;

4.ILoggerFactory

用于创建Logger对象

Log4net源码View之Logger解析第4张

其在DefaultLoggerFactory中的实现为:

  1. public Logger CreateLogger(ILoggerRepository repository, string name)
  2. {
  3.    if (name == null)
  4.    {
  5.       returnnew RootLogger(repository.LevelMap.LookupWithDefault(Level.Debug));
  6.    }
  7.    returnnew LoggerImpl(name);
  8. }

    其中,LoggerImpl对象继承自Logger,并没有做什么特殊的改变。

    RootLogger带有一个默认的Level。

总结:

Logger其实提供了一种处理中心的功能,类似分发站,将message发给各个Appender。

待续~

免责声明:文章转载自《Log4net源码View之Logger解析》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇webuploader上传文件,图片Unity中DoTween的使用下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

java面试宝典

相关概念 面向对象的三个特征 封装,继承,多态.这个应该是人人皆知.有时候也会加上抽象. 多态的好处 允许不同类对象对同一消息做出响应,即同一消息可以根据发送对象的不同而采用多种不同的行为方式(发送消息就是函数调用).主要有以下优点: 可替换性:多态对已存在代码具有可替换性. 可扩充性:增加新的子类不影响已经存在的类结构. 接口性:多态是超累通过方法签名...

smali 语言语法

Androidkiller 可以反编译Android的apk,生成一种.smali代码。(这理解好像不对) 网上找了一篇smali的语法手册,可以方便查找,文章名《Smali文件语法参考》 手册地址:http://blog.csdn.net/litton_van/article/details/7843490 网上有一篇smali的语法讲解(java与sm...

.NET链接Oracle 参数绑定问题

在.NET项目中链接Oracle使用的驱动是 Oracle.ManagedDataAccess.dll ,这里下载 所遇到的问题 使用存储过程一个参数没有问题,发现两个或两个以上会有参数没传过来的现象。 最后通过排查发现是没有添加参数绑定(问题找了好长时间,刚开始还以为驱动的问题+_+)。 需要设置设置属性 BindByName = true; 下面附上 ...

MP实战系列(十二)之封装方法详解(续二)

继续MP实战系列(十一)之封装方法详解(续一)这篇文章之后。 此次要讲的是关于查询。 查询是用的比较多的,查询很重要,好的查询,加上索引如鱼得水,不好的查询加再多索引也是无济于事。 1.selectById()方法 演示示例: UserEntity user = ud.selectById(33); System.out....

delphi计算两个时间差-转

uses DateUtils;   var S1, S2: string; T1, T2: TDateTime; D, H, M, S: Integer; Value: Int64; begin S1 := '2015/09/23 15:44:50'; S2 := '2013/09/22 16:47:51'; T1 := Str...

Spring手动绑定ConfigurationProperties

Spring加载ConfigurationProperties的典型应用是自动加载(Autowired),在一些特殊的场景需要手动修改加载的Property属性。 代码实例,config类 importjava.util.ArrayList; importjava.util.List; importorg.springframework.boot.con...