原文:log4net Tutorial

一、基础

  log4net分为三部分:配置、设置和调用。配置通常是在app.webconfig或web.config文件中;为了增加灵活性,我们也可以使用单独的配置文件。设置通常是几行代码,作用是设置和实例化一个logger连接。最后一部分是调用。

二、日志级别

  有7个日志级别,其中5个可以在代码中调用。下面是日志级别列表,按照优先级排列:

  1. OFF - nothing gets logged (cannot be called)
  2. FATAL
  3. ERROR
  4. WARN
  5. INFO
  6. DEBUG
  7. ALL - everything gets logged (cannot be called)

  这些级别会在代码和配置文件中多次使用。除了第一个和最后一个外,其他的级别没有指明代表什么含义。

三、配置

  在app.webconfig或web.config文件中配置log4net logger是标准做法。

1.Root

  我们需要一个root节点来覆盖最高级别的logger引用,这些是logger的继承信息。root节点唯一的其他作用就是定义日志的最低级别。因为所有logger均继承自root,因此低于指定级别的不会被记录为日志。这是控制程序中日志级别的简单方式,例如下面的例子中使用默认的INFO级别(这样DEBUG消息将会被忽略),并且应该启用root下面的两个appender的引用:

<root>
<level value="INFO"/>
<appender-ref ref="FileAppender"/>
<appender-ref ref="ConsoleAppender" />
</root>

2.Additional Logger

  有时我们希望知道程序特定一部分的更多信息。log4net允许我们在root logger之外指定额外的logger。例如,下面的例子中指定了一个额外的logger,记录在OtherClass类对象内的控制台日志:

<logger name="Log4NetTest.OtherClass">
<level value="DEBUG"/>
<appender-ref ref="ConsoleAppender"/>
</logger>

  需要注意的是logger name的值为包含命名空间的类名。如果我们想要监控整个命名空间,就只要使用需要监控的命名空间填充即可。我(作者)建议不要在多个logger中重复使用日志输出目的地。我们可以这样做,但会有一些 不可预知的结果。

3.CofigSections

  在一个配置文件中除了log4net外通常还会有其他配置信息,我们需要定义一个节点来标识log4net配置的位置。下面的例子,表示log4net的配置信息在“log4net”标签下面:

<configSections>
<section name="log4net"
type="log4net.Config.Log4NetConfigurationSectionHandler, log4net"/>
</configSections>

4.通用的Appender

  appender是记录信息的名字。它用来指定信息被记录在哪里,怎样被记录以及在什么情况下被记录。虽然每个appender都有不同的参数来确定信息被记录在哪里,但是还是有一些通用的元素。第一个是name和type。必须为每个appender命名,并为他分配一个类型。例如:

<appender name="ConsoleAppender" type="log4net.Appender.ConsoleAppender">

4.1.Layout:

  在每个appender内部必须有一个layout节点。尽管根据不同的appender类型会有不同,但是基础的是相同的。我们需要一个类型来指定数据如何被写入。有多个选项,但是我(作者)推荐的是使用模式布局(pattern layout)类型。因为这样允许我们指定数据如何被写入数据存储。如果我们指定模式布局类型,我们还需要一个子标签来指定转换模式(conversion pattern)。我们的数据使用这种模式写入的到数据存储。例如:

<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%date [%thread] %-5level %logger [%ndc]
- %message%newline"/>
</layout>

4.2.Conversion Pattern:

  正如上面提到的,转换模式实体是模式布局用来告诉appender如何存储信息的。在这些模式中可以使用很多不同的关键词和字符串。在这里列出我(作者)认为最有用和最重要的。所有的可以可以查看log4net文档:

  • %date - 使用本地时区信息输出日期。也可以使用花括号和布局模式格式化日期,例如%date{MMMM DD,yyyy HH:mm:ss,fff}会输出"January 01,2011 14:15:43,767"。但是建议使用log4net的日期格式(ABSOLUTE,DATE或ISO8601)这样性能会比较好。
  • %utcdate - 与%date相同,只是它输出世界时。
  • %exception - 如果传递过来一个异常,它将会被使用,并在异常信息后插入新的一行。如果没有异常传递过来,这个实体会被忽略并且不会产生新的一行。这个转换模式通常放在日志实体的最后,并且在异常前面通常也会插入一行。
  • %level - 这是我们指定的事件级别(DEBUG,INFO,WARN等)。
  • %message - 我们传递到日志事件的消息。
  • %newline - 这是新行实体。根据我们程序的平台翻译成新行字符。这是换行的首选方法,并且相对于特定的平台供应商没有性能问题。
  • %timestamp - 程序运行以来的毫秒数。
  • %thread - 实体产生的线程名(如果线程没有命名则是线程的编号)。

  下面几个也非常有用,但是使用时要特别注意,因为它们会对性能造成影响:

  • %identity - 使用Principal.Identity.Name方法获取的当前用户的用户名。
  • %location - 在调试时特别有用,它将告诉我们日志方法是否被调用(行号、方法等)。然而 ,在Release模式下信息的数量将会减少,这取决于系统访问编译后的代码。
  • %line - 代码的行号(即上面location的行号)。
  • %method - 调用日志实体的方法(即上面location的方法)。
  • %username - 输出WindowsIdentity属性的值。

  有些配置文件使用的是字母而不是名字。这有利于整个实体。我(作者)在这里不会深度介绍,我们只要知道这些实体可以被格式化为合适的宽度。可以在两边添加空格,也可以将值截断来适应内部固定宽度的列。基本的语法是将数字或值放在%号和名字之间:

  • X - 指定了字符的最小长度。如果字符的长度小于此值,将会在左侧填充空格。例如,%10message将会输出" hi"。
  • -X - 和上面相同,不同的是在右侧填充空格。例如,%-10message将会输出"hi "。
  • .X - 指定字符的最大长度。如果超出长度将会从左侧开始截取。例如,%.10message如果字符串为"Error entry"将会输出"rror entry"。

  我们可以将上面的组合使用,例如"%10.20message",如果长度不足10则在左侧填充空格,如果超过20将会截取。

4.3.Filter:

  Filter是appender的一部分。使用filter我们可以指定哪个(或哪些)级别被记录,甚至可以在消息中寻找关键字。Filter可以混合和匹配,当时当我们这样做的时候要特别注意。当消息符合filter的内部标准,它将会被记录并且filter的过程结束。这是Filter最大的缺陷,因此当我们处理复杂的filter时,filter的排序就变得非常重要。

4.3.1.StringMatchFilter:

  字符串匹配filter在被记录的信息中寻找指定的字符串。我们可以知道多个字符串匹配filter。它们在一个查询语句中类似OR组合。需要注意的是不要试图匹配一个指定的字符串不排除一个实体(因为它可以继续下一个字符串匹配filter)。这意味着,我们可能会遇到没有匹配的情况。这种情况下,默认的行为是记录这个实体。因此,在字符串匹配filter设置的最后,包含一个否认所有filter(下面会介绍)在没有匹配时组织实体被记录是非常必要的。下面是字符串匹配filter的例子:

<filter type="log4net.Filter.StringMatchFilter">
<stringToMatch value="test" />
</filter>

4.3.2.LevelRangeFilter:

  级别范围filter告诉系统只有在指定范围内的实体才会被记录。下面的例子中INFO、WARN、ERROR级别的事件会被记录,但是DEBUG事件不会被记录。我们在这个实体后面不需要定义否认所有filter,因为否定所有filter是默认的。

<filter type="log4net.Filter.LevelRangeFilter">
<levelMin value="INFO" />
<levelMax value="FATAL" />
</filter>

4.3.3.LevelMatchFilter:

  级别匹配filter与级别范围filter类似,不同的是它只捕获指定的一种级别。它没有默认的否定所有filter,因此要在级别匹配filter之后添加否定所有filter。

<filter type="log4net.Filter.LevelMatchFilter">
<levelToMatch value="ERROR"/>
</filter>

4.3.4.DenyAllFilter:

  如果被忘记,appender通常不会按照所期望的工作。指定这个条目的唯一目的是指定不产生日志实体。如果只定义了这个条目,将不会有日志产生。

<filter type="log4net.Filter.DenyAllFilter" />

5.Appender

  每个类型的appender都有它自己的语法定义数据存储到哪里。最不寻常的是把日志存储到数据库。

5.1.Console Appender:

  我(作者)通常使用这个appender来测试,但是它在产品上同样有用。它写入到输出窗口,如果我们使用的是console程序它会写入到命令窗口。下面的filter输出的值类似"2010-12-26 15:41:03,581 [10] WARN Log4NetTest.frmMain - This is a WARN test.",在它的末尾将会换行。

<appender name="ConsoleAppender" type="log4net.Appender.ConsoleAppender">
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%date{ABSOLUTE}
[%thread] %level %logger - %message%newline"/>
</layout>
<filter type="log4net.Filter.StringMatchFilter">
<stringToMatch value="test" />
</filter>
<filter type="log4net.Filter.DenyAllFilter" />
</appender>

5.2.File Appender:

  这个appender将会写入一个文本文件。该appender必须指定文件名(下面的例子中文件名为mylogfile.txt,将会存储在与可执行文件相同的路径),我们指定在文件末尾添加(而不是覆盖),并且指定FileAppender使用最小锁这样该文件可以被多个appender使用。

<appender name="FileAppender" type="log4net.Appender.FileAppender">
<file value="mylogfile.txt" />
<appendToFile value="true" />
<lockingModel type="log4net.Appender.FileAppender+MinimalLock" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%date [%thread] %level %logger - %message%newline" />
</layout>
<filter type="log4net.Filter.LevelRangeFilter">
<levelMin value="INFO" />
<levelMax value="FATAL" />
</filter>
</appender>

5.3.Rolling File Appender:

  应该尽量使用该Appender代替file appender。Rolling file appender的目的是与file appender执行相同的功能,但是每个文件只存储一定量的数据。这样就不用担心日志耗尽存储空间。如果没有使用滚动选项,在足够时间的情况下,即使是一个小的应用程序也会耗尽存储空间。在下面的例子中使用于上面例子类似的方式记录日志,但是指定了每个文件的大小为10MB,并且在删除之前有5个存档文件(从最之前的文件开始删除)。存档文件具有相同的名字,只是在每个文件名字后添加一个点和数组(例如,第二个存档的文件名为mylogfile.txt.2).staticLogFileName确保日志文件与指定的文件名相同(本例中为mylogfile.txt)。

<appender name="RollingFileAppender" type="log4net.Appender.RollingFileAppender">
<file value="mylogfile.txt" />
<appendToFile value="true" />
<rollingStyle value="Size" />
<maxSizeRollBackups value="5" />
<maximumFileSize value="10MB" />
<staticLogFileName value="true" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%date [%thread] %level %logger - %message%newline" />
</layout>
</appender>

5.4.ADO.NET Appender:

  下面的例子是写入SQL Server数据库,但是我们可以使用这种模式写入任意数据库。

  注意:如果我们发现ADO.NET appender没有反应,查看bufferSize的值。这个值定义的是将日志写入数据库之前log4
net缓存的语句数量。

  更多信息请参考:http://logging.apache.org/log4net/release/config-examples.html

<appender name="AdoNetAppender" type="log4net.Appender.AdoNetAppender">
<bufferSize value="100" />
<connectionType value="System.Data.SqlClient.SqlConnection,
System.Data, Version=1.0.3300.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" />
<connectionString value="data source=[database server];
initial catalog=[database name];integrated security=false;
persist security info=True;User ID=[user];Password=[password]" />
<commandText value="INSERT INTO Log ([Date],[Thread],[Level],[Logger],
[Message],[Exception]) VALUES (@log_date, @thread, @log_level,
@logger, @message, @exception)" />
<parameter>
<parameterName value="@log_date" />
<dbType value="DateTime" />
<layout type="log4net.Layout.RawTimeStampLayout" />
</parameter>
<parameter>
<parameterName value="@thread" />
<dbType value="String" />
<size value="255" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%thread" />
</layout>
</parameter>
<parameter>
<parameterName value="@log_level" />
<dbType value="String" />
<size value="50" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%level" />
</layout>
</parameter>
<parameter>
<parameterName value="@logger" />
<dbType value="String" />
<size value="255" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%logger" />
</layout>
</parameter>
<parameter>
<parameterName value="@message" />
<dbType value="String" />
<size value="4000" />
<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%message" />
</layout>
</parameter>
<parameter>
<parameterName value="@exception" />
<dbType value="String" />
<size value="2000" />
<layout type="log4net.Layout.ExceptionLayout" />
</parameter>
</appender>

四、代码

  在我们的程序中引入log4net的dll时,有三行代码是我们需要知道的。第一个是需要一次性放置在我们的类中。我(作者)通常它放在Program.cs文件的using语句下面:

[assembly: log4net.Config.XmlConfigurator(Watch = true)]

  第二个是需要在每个类中用到的,定义一个变量用来调用log4net的方法。它使用System.Reflection调用后去当前类的信息:

private static readonly log4net.ILog log = log4net.LogManager.GetLogger
(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);

  最后的代码块是实际调用来记录一些信息。例如:

log.Info("Info logging");

  注意,我们可以在后面添加一个可选参数包含异常信息。与下面的非常类似:

log.Error("This is my error", ex);

  ex是异常对象。我们需要使用%exception来捕获异常信息。

五、记录额外的数据

  通常使用log4net的基本配置就可以为应用程序提供足够的信息。但是,有时我们想要使用标准方法记录更多的信息。例如,在ADO.NET appender中,我们可能想要添加用户名字段而不仅仅是信息字段。没有一种转换模式与应用程序中的用户名匹配。然而,我们可以使用上下文属性来自定义可以被appender访问的属性。例如:

log4net.GlobalContext.Properties["testProperty"] = "This is my test property information";

  需要注意几点:1.自定义属性可以任意命名,但是如果命名与已经存在的命名相同,将会把原来的覆盖。2.本例中使用的是Global上下文,但是有四个上下文可以被利用。它们是基于线程的。Global定义的可以应用在程序进程、逻辑继承和进一步限制时间范围的任何地方。如果两个属性具有相同的名字,将会取窄范围的那个的值。

  捕获自定义属性的例子:

<layout type="log4net.Layout.PatternLayout">
<conversionPattern value="%date{ABSOLUTE} [%thread] %level
%logger - %message%newlineExtra Info: %property{
testProperty}%newline%exception"/>
</layout>

  更多信息请查看:http://logging.apache.org/log4net/release/manual/contexts.html

六、不使用app.config/web.config

  有时我们需要单独的文件存储log4net的配置信息。实际上这是存储配置信息最好的方法,因为我们可以有不同的配置副本应用于项目。这可以减少开发时间并且允许我们记录标准化的日志信息。如果要单独存储配置文件,我们需要更改程序的两个地方。第一件是将配置文件保存在另一个文件。第二件是修改调用设置,例如:

[assembly: log4net.Config.XmlConfigurator(ConfigFile =
"MyStandardLog4Net.config", Watch = true)]  

  还可以使用另一种方法,这种方法需要把配置文件命名为程序集名称(包括扩展名):

[assembly: log4net.Config.XmlConfigurator(ConfigFileExtension = "mylogger", Watch = true)]

  在上面的例子中如果我们的程序是test.exe,log4net的配置文件需要命名为test.exe.mylogger。

七、配置文件模板

<!--This is the root of your config file-->
<configuration> <!-- Level 0 -->
<!--This specifies what the section name is-->
<configSections> <!-- Level 1 -->
<section name="log4net"
type="log4net.Config.Log4NetConfigurationSectionHandler,
log4net"/> <!-- Level 2 -->
</configSections>
<log4net> <!-- Level 1 -->
<appender> <!-- Level 2 -->
<layout> <!-- Level 3 -->
<conversionPattern /> <!-- Level 4 -->
</layout>
<filter> <!-- Level 3 -->
</filter>
</appender>
<root> <!-- Level 2 -->
<level /> <!-- Level 3 -->
<appender-ref /> <!-- Level 3 -->
</root>
<logger> <!-- Level 2 -->
<level /> <!-- Level 3 -->
<appender-ref /> <!-- Level 3 -->
</logger>
</log4net>
</configuration>

  发现已经有园友翻译过这篇文章,而且比我翻译的好很多:Log4Net指南

[翻译]log4net教程的更多相关文章

  1. [翻译svg教程]svg学习系列 开篇

    目录 [翻译svg教程]svg学习系列 开篇 [翻译svg教程 ]svg 的坐标系统 [翻译svg教程]svg 中的g元素 [翻译svg教程]svg中矩形元素 rect [翻译svg教程]svg中的c ...

  2. Entity Framework教程翻译 ---- 系列教程

    Entity Framework教程(第二版) (翻译)Entity Framework技巧系列之十四 - Tip 56 (翻译)Entity Framework技巧系列之十三 - Tip 51 - ...

  3. 《Entity Framework 6 Recipes》中文翻译 ---- 系列教程

    为了方便大家的阅读和学习,也是响应网友的建议,在这里为这个系列做一个目录.在目录开始这前,我先来回答之前遇到的几个问题. 1.为什么要学习EF? 这个问题很简单,项目需要.这不像学校,没人强迫你学习! ...

  4. [翻译svg教程]Path元素 svg中最神奇的元素!

    先看一个实例 <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999 ...

  5. [翻译svg教程]svg中的circle元素

    svg中的<circle> 元素,是用来绘制圆形的,例如 <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink= ...

  6. [翻译svg教程]svg中矩形元素 rect

    svg 元素<rect> 是一个矩形元素,用这个元素,可以你可以绘制矩形,设置矩形宽高,边框的宽度颜色,矩形的填充颜色,是否用圆角等 rect 示例 <svg xmlns=" ...

  7. [翻译svg教程]svg 中的g元素

    svg 中的<g>元素用来组织svg元素.如果一组svg元素被g元素包裹了,你可以通过对g元素进行变换(transform),被g元素包裹的元素也将被变换,就好这些被svg包裹的元素是一个 ...

  8. 【翻译svg教程 】svg 的坐标系统

    http://tutorials.jenkov.com/svg/svg-coordinate-system.html svg的坐标系统(和大多数计算机绘图的坐标系统)和数学中绘图系统有点不一样 数学/ ...

  9. LOG4NET图文教程

    LOG4NET教程 一:简介 从操作系统到大多数的大型软件,都会有自己的程序运行时的日志跟踪API.因为一旦程序被部署以后,就不太可能再利用专门的调试工具了.然而软件开发人员需要一套强大的日志系统来记 ...

随机推荐

  1. Ant-常用命令(笔记二)

    1.copy命令: <?xml version="1.0" encoding="utf-8"?> <project name="co ...

  2. 怎样通过iPhone Safari 来安装测试版ipa

    http://www.cocoachina.com/bbs/read.php?tid=94101# <?xml version="1.0" encoding="UT ...

  3. Java Singleton 单例模式

    大家可能还听过 Singleton  也就是单例模式 这个单例模式要求 在程序的运行时候   一个程序的某个类 只允许产生一个 实例 那么 这个类就是一个单例类 Java Singleton模式主要作 ...

  4. SQLyog破解版:SQLyog MySQL GUI 11.2.4-0 Ultimate中文版 带序列号【转载】

    SQLyog 是一个易于使用的.快速而简洁的图形化管理MYSQL数据库的工具,目前(2013年9月11日)最新版为:SQLyog Ultimate – MySQL GUI v11.24,本站已亲测比较 ...

  5. Upcase 将edit1中的每个字符串改为首字母大写

    //将edit1中的每个字符串改为首字母大写 procedure TForm1.Edit1KeyPress(Sender: TObject; var Key: Char);begin    with ...

  6. (转)如何在JavaScript与ActiveX之间传递数据2

    本文研究如何在JS等脚本语言与ActiveX控件之间通信,如何传递各种类型的参数,以及COM的IDispatch接口.使用类似的方法,可以推广到其他所有脚本型语言,如LUA,AutoCad等.本文将研 ...

  7. MySQL的数据类型(转)

    MySQL的数据类型 1.整数 TINYINT: 8 bit 存储空间 SMALLINT: 16 bit 存储空间 MEDIUMINT: 24 bit 存储空间 INT: 32 bit 存储空间 BI ...

  8. java_jdbc_可滚动结果集与分页

    public static void create2(int i) { Connection conn = null; Statement st = null; ResultSet rs = null ...

  9. iOS 中使用.9图

    背景 .9图来源于Android.为了设计出一套图,兼容Android和iOS,使用.9图的方式来对图片进行拉伸以适应不同的屏幕.在iOS中没有.9图的概念,只能先了解Android的.9图再进行模拟 ...

  10. 学习opencv中文版教程——第二章

    学习opencv中文版教程——第二章 所有案例,跑起来~~~然而并没有都跑起来...我只把我能跑的都尽量跑了,毕竟看书还是很生硬,能运行能出结果,才比较好. 越着急,心越慌,越是着急,越要慢,越是陌生 ...