1.简介

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

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

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

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

所以,从下面开始,请先忽略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. virtual protected void 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. public struct LoggingEventData
  2. {
  3.    public string LoggerName;
  4.    public Level Level;
  5.    public string Message;
  6.    public string ThreadName;
  7.    public DateTime TimeStamp;
  8.    public LocationInfo LocationInfo;
  9.    public string UserName;
  10.    public string Identity;
  11.    public string ExceptionString;
  12.    public string Domain;
  13.    public PropertiesDictionary Properties;
  14. }

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

3.抽象类Logger

特点:

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

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

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

4.ILoggerFactory

用于创建Logger对象

其在DefaultLoggerFactory中的实现为:

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

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

RootLogger带有一个默认的Level。

总结:

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

待续~

Log4net源码View之Logger解析的更多相关文章

  1. FFmpeg的HEVC解码器源码简单分析:解析器(Parser)部分

    ===================================================== HEVC源码分析文章列表: [解码 -libavcodec HEVC 解码器] FFmpeg ...

  2. jQuery 2.0.3 源码分析Sizzle引擎解析原理

    jQuery 2.0.3 源码分析Sizzle引擎 - 解析原理 声明:本文为原创文章,如需转载,请注明来源并保留原文链接Aaron,谢谢! 先来回答博友的提问: 如何解析 div > p + ...

  3. springMVC源码分析--RequestParamMethodArgumentResolver参数解析器(三)

    之前两篇博客springMVC源码分析--HandlerMethodArgumentResolver参数解析器(一)和springMVC源码解析--HandlerMethodArgumentResol ...

  4. MyBatis 源码分析 - 映射文件解析过程

    1.简介 在上一篇文章中,我详细分析了 MyBatis 配置文件的解析过程.由于上一篇文章的篇幅比较大,加之映射文件解析过程也比较复杂的原因.所以我将映射文件解析过程的分析内容从上一篇文章中抽取出来, ...

  5. Appium Android Bootstrap源码分析之命令解析执行

    通过上一篇文章<Appium Android Bootstrap源码分析之控件AndroidElement>我们知道了Appium从pc端发送过来的命令如果是控件相关的话,最终目标控件在b ...

  6. Spring源码:IOC原理解析(二)

    版权声明:本文为博主原创文章,转载请注明出处,欢迎交流学习! 接着上一章节的内容,我们来分析当new一个FileSystemXmlApplicationContext对象的时候,spring到底做了那 ...

  7. Spring mybatis源码篇章-MybatisDAO文件解析(二)

    前言:通过阅读源码对实现机制进行了解有利于陶冶情操,承接前文Spring mybatis源码篇章-MybatisDAO文件解析(一) 默认加载mybatis主文件方式 XMLConfigBuilder ...

  8. Spring mybatis源码篇章-MybatisDAO文件解析(一)

    前言:通过阅读源码对实现机制进行了解有利于陶冶情操,承接前文Spring mybatis源码篇章-SqlSessionFactory 加载指定的mybatis主文件 Mybatis模板文件,其中的属性 ...

  9. springMVC源码分析--ViewResolver视图解析器(一)

    SpringMVC用于处理视图最重要的两个接口是ViewResolver和View.ViewResolver的主要作用是把一个逻辑上的视图名称解析为一个真正的视图,SpringMVC中用于把View对 ...

随机推荐

  1. 如何在Linux命令行中创建以及展示演示稿

    导读 你在准备一场演讲的时候,脑海可能会先被图文并茂.形象华丽的演示图稿所占据.诚然,没有人会否认一份生动形象的演讲稿所带来的积极作用.然而,并非所有的演讲都需要TED Talk的质量.更多时候,演讲 ...

  2. 服务器部署之 cap deploy:setup

    文章是从我的个人博客上粘贴过来的, 大家也可以访问 www.iwangzheng.com $ cap deploy:setup 执行到这一步的时候会时间较长,可以直接中断 * executing &q ...

  3. Decompiled .class file,bytecode version:51.0(Java 7) Source for 'Android API 23 Platform' not found

    今天在Android Studio中访问Java源码的时候,代码上方出现如下提示: 而且方法体中显示介样内容: throw new RuntimeException("Stub!" ...

  4. The Perfect Stall (incomplete)

    恩,一看就知道是一道二分图最大匹配的题. 感动得发现自己不会做..果然我是太弱了.学校里真是麻烦死,根本没有时间好吗. (NOIP)会不会感动地滚粗啊? 然后稍微看看,恩,匈牙利算法. 真是感动得落泪 ...

  5. Linux awk命令详解??????????(研究)

    http://blog.chinaunix.net/uid-25120309-id-3801250.html 一.  AWK 说明  awk是一种编程语言,用于在linux/unix下对文本和数据进行 ...

  6. 【OpenStack】OpenStack系列1之Python虚拟环境搭建

    安装virtualenv相关软件包 安装:yum install python-virtualenv* -y 简介,安装包主要包括, python-virtualenv:virtualenv用于创建独 ...

  7. mysql 恢复备份

    1.在my.cnf 文件中增加以下配置 log-bin=/var/lib/mysql/mysql-binloglog-bin-index = /var/lib/mysql/mysql-binlog 2 ...

  8. poj 2739 Sum of Consecutive Prime Numbers 解题报告

    题目链接:http://poj.org/problem?id=2739 预处理出所有10001以内的素数,按照递增顺序存入数组prime[1...total].然后依次处理每个测试数据.采用双重循环计 ...

  9. JDK JRE 区别

    JDK  包含了编译器,比如让.java编译成.classs文件. JRE =Java Runtime Environment j是一些比如一些split函数需要的包,都在里面,基本的运行环境都在JR ...

  10. 【读书笔记】读《JavaScript模式》 - 对象创建模式

    JavaScript是一种简洁明了的语言,其中并没有在其他语言中经常使用的一些特殊语法特征,比如命名空间(namespace).模块(module).包(package).私有属性(private p ...