【转】使用Log4Net进行日志记录

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 首先说说为什么要进行日志记录。在一个完整的程序系统里面,日志系统是一个非常重要的功能组成部分。它可以记录下系统所产生的所有行为,并按照某种规范表达出来。我们可以使用日志系统所记录的信息为系统进行排错,优化系统的性能,或者根据这些信息调整系统的行为。

     首先说说为什么要进行日志记录。在一个完整的程序系统里面,日志系统是一个非常重要的功能组成部分。它可以记录下系统所产生的所有行为,并按照某种规范表达出来。我们可以使用日志系统所记录的信息为系统进行排错,优化系统的性能,或者根据这些信息调整系统的行为。

     Log4net是一个很著名的开源的日志记录组件。官方网址为:http://logging.apache.org/log4net/ ,使用Log4net能够很简单的为我们的程序添加日志记录功能。下面我们先通过一个网站例子来说明如何在.net中使用log4net。

创建日志记录步骤

第一步,

       当然是添加log4net.dll的引用啦,我这里提供一个log4net的dll文件,懒得去官网下的可以到这里下http://www.vdisk.cn/down/index/7509396A7366

第二步,

       在AssemblyInfo.cs文件中添加下面一句话:

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

这句话的意思是log4net会自动寻找配置文件App.config或Web.config从而获得并加载其中的配置信息。如果想log4net随时监视配置文件以便重新加载的话就要这样写按照上面一样写(winform程序ConfigFile为App.config)。

第三步,

      配置Web.config。

<configuration>

  <configSections>

        <!--注意:这里需要添加--> 
        <section name="log4net" type="System.Configuration.IgnoreSectionHandler"/> 
   </configSections> 
  <connectionStrings> 
    <add name="ApplicationServices" 
         connectionString="data source=.\SQLEXPRESS;Integrated Security=SSPI;AttachDBFilename=|DataDirectory|\aspnetdb.mdf;User Instance=true" 
         providerName="System.Data.SqlClient" /> 
  </connectionStrings>

  

   <!--注意:这里需要添加-->

   <log4net>


        <!--定义输出到文件中--> 
        <appender name="LogFileAppender" type="log4net.Appender.FileAppender"> 
            <!--定义文件存放位置--> 
            <file value="C:\log.txt" /> 
            <appendToFile value="true" /> 
            <rollingStyle value="Date" /> 
            <datePattern value="yyyyMMdd-HH:mm:ss" /> 
            <layout type="log4net.Layout.PatternLayout"> 
                <!--输出格式--> 
                <!--样例:2008-03-26 13:42:32,111 [10] INFO  Log4NetDemo.MainClass [(null)] - info--> 
                <conversionPattern value="记录时间:%date 线程ID:[%thread] 日志级别:%-5level 出错类:%logger property:[%property{NDC}] - 错误描述:%message%newline" /> 
            </layout> 
        </appender>


        <!--定义输出到控制台命令行中--> 
        <appender name="ConsoleAppender" type="log4net.Appender.ConsoleAppender"> 
            <layout type="log4net.Layout.PatternLayout"> 
                <conversionPattern value="%date [%thread] %-5level %logger [%property{NDC}] - %message%newline" /> 
            </layout> 
        </appender>


        <!--定义输出到windows事件中--> 
        <appender name="EventLogAppender" type="log4net.Appender.EventLogAppender"> 
            <layout type="log4net.Layout.PatternLayout"> 
                <conversionPattern value="%date [%thread] %-5level %logger [%property{NDC}] - %message%newline" /> 
            </layout> 
        </appender>


        <!--定义输出到数据库中,这里举例输出到Access数据库中,数据库为C盘的log4net.mdb--> 
        <appender name="AdoNetAppender_Access" type="log4net.Appender.AdoNetAppender"> 
            <connectionString value="Provider=Microsoft.Jet.OLEDB.4.0;Data Source=C:log4net.mdb" /> 
            <commandText value="INSERT INTO LogDetails ([LogDate],[Thread],[Level],[Logger],[Message]) VALUES (@logDate, @thread, @logLevel, @logger,@message)" /> 
            <!--定义各个参数--> 
            <parameter> 
                <parameterName value="@logDate" /> 
                <dbType value="String" /> 
                <size value="240" /> 
                <layout type="log4net.Layout.PatternLayout"> 
                    <conversionPattern value="%date" /> 
                </layout> 
            </parameter> 
            <parameter> 
                <parameterName value="@thread" /> 
                <dbType value="String" /> 
                <size value="240" /> 
                <layout type="log4net.Layout.PatternLayout"> 
                    <conversionPattern value="%thread" /> 
                </layout> 
            </parameter> 
            <parameter> 
                <parameterName value="@logLevel" /> 
                <dbType value="String" /> 
                <size value="240" /> 
                <layout type="log4net.Layout.PatternLayout"> 
                    <conversionPattern value="%level" /> 
                </layout> 
            </parameter> 
            <parameter> 
                <parameterName value="@logger" /> 
                <dbType value="String" /> 
                <size value="240" /> 
                <layout type="log4net.Layout.PatternLayout"> 
                    <conversionPattern value="%logger" /> 
                </layout> 
            </parameter> 
            <parameter> 
                <parameterName value="@message" /> 
                <dbType value="String" /> 
                <size value="240" /> 
                <layout type="log4net.Layout.PatternLayout"> 
                    <conversionPattern value="%message" /> 
                </layout> 
            </parameter> 
        </appender>


        <!--定义日志的输出媒介,下面定义日志以四种方式输出。也可以下面的按照一种类型或其他类型输出。--> 
        <root> 
            <!--文件形式记录日志--> 
            <appender-ref ref="LogFileAppender" /> 
            <!--控制台控制显示日志--> 
            <appender-ref ref="ConsoleAppender" /> 
            <!--Windows事件日志--> 
            <!--<appender-ref ref="EventLogAppender" />--> 
            <!-- 如果不启用相应的日志记录,可以通过这种方式注释掉 
                <appender-ref ref="AdoNetAppender_Access" /> 
            --> 
        </root>

    </log4net>


  <system.web> 
    <compilation debug="true" targetFramework="4.0" /> 
    <httpModules>

      <add name="fileUpload" type="WebApplication3.ProcessFileModule"/> 
    </httpModules> 
    <authentication mode="Forms"> 
      <forms loginUrl="~/Account/Login.aspx" timeout="2880" /> 
    </authentication>

    <membership> 
      <providers> 
        <clear/> 
        <add name="AspNetSqlMembershipProvider" type="System.Web.Security.SqlMembershipProvider" connectionStringName="ApplicationServices" 
             enablePasswordRetrieval="false" enablePasswordReset="true" requiresQuestionAndAnswer="false" requiresUniqueEmail="false" 
             maxInvalidPasswordAttempts="5" minRequiredPasswordLength="6" minRequiredNonalphanumericCharacters="0" passwordAttemptWindow="10" 
             applicationName="/" /> 
      </providers> 
    </membership>

    <profile> 
      <providers> 
        <clear/> 
        <add name="AspNetSqlProfileProvider" type="System.Web.Profile.SqlProfileProvider" connectionStringName="ApplicationServices" applicationName="/"/> 
      </providers> 
    </profile>

    <roleManager enabled="false"> 
      <providers> 
        <clear/> 
        <add name="AspNetSqlRoleProvider" type="System.Web.Security.SqlRoleProvider" connectionStringName="ApplicationServices" applicationName="/" /> 
        <add name="AspNetWindowsTokenRoleProvider" type="System.Web.Security.WindowsTokenRoleProvider" applicationName="/" /> 
      </providers> 
    </roleManager> 
    <httpRuntime maxRequestLength="10240000"/> 
  </system.web>

  <system.webServer> 
    <modules runAllManagedModulesForAllRequests="true"/> 
  </system.webServer>

</configuration>

上面这段配置取自周公,配置的说明上面已经注释的比较详细了。

第四步,

在程序中记录信息。我们在项目下Default.aspx的Page_Load方法加入如下代码:

           //获得log4net实例 
           ILog log = log4net.LogManager.GetLogger("LogFileAppender"); 
           log.Info("Default.aspx下面的PageLoad方法被执行");

然后运行项目,可以发现在c:\log4netfile.txt中已经记录了一条日志信息:

记录时间:2011-04-21 21:26:17,729 线程ID:[4] 日志级别:INFO  出错类:LogFileAppender property:[(null)] - 错误描述:Default.aspx下面的PageLoad方法被执行

通过上面这几步我想您已经能够为程序创建一些简单日志记录功能了。下面我们再来详细说说log4net其他的一些使用方法与特点。

log4net详细说明

    这里我通过问答的形式来说明log4net的一些应用。

1.log4net总共有几种记录方式,每种方式该如何配置与使用?

Log4net目前支持的输出方式包括:

1 AdoNetAppender

     将日志记录到数据库中。可以采用SQL和存储过程两种方式。

2 AnsiColorTerminalAppender

     在ANSI 窗口终端写下高亮度的日志事件。

3 AspNetTraceAppender

     能用asp.net中Trace的方式查看记录的日志。

4 BufferingForwardingAppender

     在输出到子Appenders之前先缓存日志事件。

5 ConsoleAppender

     将日志输出到控制台。

6 EventLogAppender

     将日志写到Windows Event Log.

7 FileAppender

     将日志写到文件中。

8 LocalSyslogAppender

     将日志写到local syslog service (仅用于UNIX环境下).

9 MemoryAppender

     将日志存到内存缓冲区。

10 NetSendAppender

     将日志输出到Windows Messenger service.这些日志信息将在用户终端的对话框中显示。

11 RemoteSyslogAppender

     通过UDP网络协议将日志写到Remote syslog service。

12 RemotingAppender

     通过.NET Remoting将日志写到远程接收端。

13 RollingFileAppender

     将日志以回滚文件的形式写到文件中。

14 SmtpAppender

     将日志写到邮件中。

15 TraceAppender

     将日志写到.NET trace 系统。

16 UdpAppender

     将日志connectionless UDP datagrams的形式送到远程宿主或以UdpClient的形式广播。

 

可以看到目前支持的方式还是很多的,我这里调几个认为常用的做个例子。

  1.文件的方式我们上面已经讲过了,这里不再做例子了。

  2.数据库方式:

    首先,添加数据库记录appender

<appender name="AdoNetAppender_Access" type="log4net.Appender.AdoNetAppender"> 
        <!--数据库连接字符串--> 
        <connectionString value="Provider=Microsoft.Jet.OLEDB.4.0;Data Source=C:log4net.mdb" /> 
        <commandText value="INSERT INTO LogDetails ([LogDate],[Thread],[Level],[Logger],[Message]) VALUES (@logDate, @thread, @logLevel, @logger,@message)" /> 
        <!--定义各个参数--> 
        <parameter> 
            <parameterName value="@logDate" /> 
            <dbType value="String" /> 
            <size value="240" /> 
            <layout type="log4net.Layout.PatternLayout"> 
                <conversionPattern value="%date" /> 
            </layout> 
        </parameter> 
        <parameter> 
            <parameterName value="@thread" /> 
            <dbType value="String" /> 
            <size value="240" /> 
            <layout type="log4net.Layout.PatternLayout"> 
                <conversionPattern value="%thread" /> 
            </layout> 
        </parameter> 
        <parameter> 
            <parameterName value="@logLevel" /> 
            <dbType value="String" /> 
            <size value="240" /> 
            <layout type="log4net.Layout.PatternLayout"> 
                <conversionPattern value="%level" /> 
            </layout> 
        </parameter> 
        <parameter> 
            <parameterName value="@logger" /> 
            <dbType value="String" /> 
            <size value="240" /> 
            <layout type="log4net.Layout.PatternLayout"> 
                <conversionPattern value="%logger" /> 
            </layout> 
        </parameter> 
        <parameter> 
            <parameterName value="@message" /> 
            <dbType value="String" /> 
            <size value="240" /> 
            <layout type="log4net.Layout.PatternLayout"> 
                <conversionPattern value="%message" /> 
            </layout> 
        </parameter> 
    </appender>

然后建立c:log4net.mdb 数据库,并在代码中启用数据库记录方式:

           //获得log4net实例 
           ILog log = log4net.LogManager.GetLogger("AdoNetAppender_Access"); 
           log.Debug("Default.aspx下面的PageLoad方法被执行");

这样日志就被记录到数据库中了。

2.log4net如何过滤我想要的日志信息,比如一个日志中只记录程序错误日志?

这个需求可以通过配置filter来实现。具体操作如下:

    全局方式

    这种方式会将级别应用于所有的日志输入方式。具体操作为在root节点下添加:

     <!--控制级别,由低到高: ALL|DEBUG|INFO|WARN|ERROR|FATAL|OFF 
         比如定义级别为INFO,则INFO级别向下的级别,比如DEBUG日志将不会被记录 
     --> 
     <!--如果没有定义LEVEL的值,则缺省为DEBUG--> 
     <level value="ERROR" />

这样对于所有的日志记录方式,所有地域ERROR级别的都不会被记录了。

     单独配置级别方式:

      该方式不会影响其他的日志输入方式。方法如下为,在具体的appender下添加filter:

          <filter type="log4net.Filter.LevelRangeFilter"> 
               <param name="LevelMin" value="ERROR" /> 
               <param name="LevelMax" value="Fatal" /> 
           </filter>

这种方式可以配置记录级别的方位,如果只需要记录一种,则最大和最小设置一样就行了。

3.你上面的日志输出格式我不喜欢,如何在log4net中配置我想要的日志输出格式?

log4net的输入格式定义在每个appender的layout中。具体的有以下内置可选项:

%m(message):输出的日志消息,如ILog.Debug(…)输出的一条消息 
%n(new line):换行 
%d(datetime):输出当前语句运行的时刻 
%r(run time):输出程序从运行到执行到当前语句时消耗的毫秒数 
%t(thread id):当前语句所在的线程ID 
%p(priority): 日志的当前优先级别,即DEBUG、INFO、WARN…等 
%c(class):当前日志对象的名称,例如: 
%f(file):输出语句所在的文件名。 
%l(line):输出语句所在的行号。 
%数字:表示该项的最小长度,如果不够,则用空格填充,如“%-5level”表示level的最小宽度是5个字符,如果实际长度不够5个字符则以空格填充。

通过这些东西,你可以任意组合你喜欢的输出格式内容。

 

4.有没有方式控制程序自动按日期记录日志信息,即每天的日志都在不同的日志文件中?

其实这种方式对应于一种特定的记录方式:RollingFileAppender。这种方式也是基于文件记录的,不过他提供更加灵活的日志记录方式。具体说来他可以按日志文件的大小或者日志记录的时间进行自动变换日志文件。

按每天不同的日期进行记录分类:

<appender name="RollingLogFileAppenderEveryMin" type="log4net.Appender.RollingFileAppender">

<!--日志文件名开头-->

<file value="c:\logfile.txt" />

<!--是否追加到文件-->

<appendToFile value="true" />

<!--变换的形式为日期-->

<rollingStyle value="Date" />

<!--日期的格式,每分钟换一个文件记录-->

<datePattern value="yyyyMMdd-HHmm" />

<layout type="log4net.Layout.PatternLayout">

<conversionPattern value="%date [%thread] %-5level %logger [%property{NDC}] - %message%newline" />

</layout>

</appender>

按照日志文件的大小进行变换,通过这种方式可以有效降低日志文件体积膨胀的问题:

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

<!--日志文件名-->

<file value="c:\log.txt"/>

<!--是否在文件中追加-->

<appendToFile value="true"/>

<!--按照文件的大小进行变换日志文件-->

<rollingStyle value="Size"/>

<!--最大变换数量,如果超过这个数量则从第一个文件开始复写-->

<maxSizeRollBackups value="10"/>

<!--最大文件大小,支持KB,MB,GB-->

<maximumFileSize value="100KB"/>

<!--日志文件名是否为静态-->

<staticLogFileName value="true"/>

<layout type="log4net.Layout.PatternLayout">

<conversionPattern value="%date [%thread] %-5level %logger [%property{NDC}] - %message%newline"/>

</layout>

</appender>

最后从网上找了一段写日志的原则,感觉还是很好的:

【写日志的原则】

Ⅰ.在catch后,把异常写入日志.

Ⅱ.在调用第三方控件的开始和结束处.

Ⅲ.在连接数据库的开始结束处.

Ⅳ.除非必要,不要在循环体中加入日志,否则一旦出问题可能导致日志暴增.

Ⅴ.在自己认为很重要的逻辑处写入日志.

 

如果要将log方法写在类库里面,那么可以参考这篇文章http://hi.baidu.com/sjbh/blog/item/10cda8d689fb0c3807088b87.html

 

 

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
目录
相关文章
|
2月前
|
XML 安全 Java
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
本文介绍了Java日志框架的基本概念和使用方法,重点讨论了SLF4J、Log4j、Logback和Log4j2之间的关系及其性能对比。SLF4J作为一个日志抽象层,允许开发者使用统一的日志接口,而Log4j、Logback和Log4j2则是具体的日志实现框架。Log4j2在性能上优于Logback,推荐在新项目中使用。文章还详细说明了如何在Spring Boot项目中配置Log4j2和Logback,以及如何使用Lombok简化日志记录。最后,提供了一些日志配置的最佳实践,包括滚动日志、统一日志格式和提高日志性能的方法。
433 30
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
|
23天前
|
监控 安全 Apache
什么是Apache日志?为什么Apache日志分析很重要?
Apache是全球广泛使用的Web服务器软件,支持超过30%的活跃网站。它通过接收和处理HTTP请求,与后端服务器通信,返回响应并记录日志,确保网页请求的快速准确处理。Apache日志分为访问日志和错误日志,对提升用户体验、保障安全及优化性能至关重要。EventLog Analyzer等工具可有效管理和分析这些日志,增强Web服务的安全性和可靠性。
|
3月前
|
XML JSON Java
Logback 与 log4j2 性能对比:谁才是日志框架的性能王者?
【10月更文挑战第5天】在Java开发中,日志框架是不可或缺的工具,它们帮助我们记录系统运行时的信息、警告和错误,对于开发人员来说至关重要。在众多日志框架中,Logback和log4j2以其卓越的性能和丰富的功能脱颖而出,成为开发者们的首选。本文将深入探讨Logback与log4j2在性能方面的对比,通过详细的分析和实例,帮助大家理解两者之间的性能差异,以便在实际项目中做出更明智的选择。
357 3
|
1天前
|
SQL 关系型数据库 MySQL
MySQL事务日志-Undo Log工作原理分析
事务的持久性是交由Redo Log来保证,原子性则是交由Undo Log来保证。如果事务中的SQL执行到一半出现错误,需要把前面已经执行过的SQL撤销以达到原子性的目的,这个过程也叫做"回滚",所以Undo Log也叫回滚日志。
MySQL事务日志-Undo Log工作原理分析
|
5天前
|
JSON 安全 API
.net 自定义日志类
在.NET中,创建自定义日志类有助于更好地管理日志信息。示例展示了如何创建、配置和使用日志记录功能,包括写入日志文件、设置日志级别、格式化消息等。注意事项涵盖时间戳、日志级别、JSON序列化、线程安全、日志格式、文件处理及示例使用。请根据需求调整代码。
29 13
|
1月前
|
存储 监控 安全
什么是事件日志管理系统?事件日志管理系统有哪些用处?
事件日志管理系统是IT安全的重要工具,用于集中收集、分析和解释来自组织IT基础设施各组件的事件日志,如防火墙、路由器、交换机等,帮助提升网络安全、实现主动威胁检测和促进合规性。系统支持多种日志类型,包括Windows事件日志、Syslog日志和应用程序日志,通过实时监测、告警及可视化分析,为企业提供强大的安全保障。然而,实施过程中也面临数据量大、日志管理和分析复杂等挑战。EventLog Analyzer作为一款高效工具,不仅提供实时监测与告警、可视化分析和报告功能,还支持多种合规性报告,帮助企业克服挑战,提升网络安全水平。
|
3月前
|
存储 缓存 关系型数据库
MySQL事务日志-Redo Log工作原理分析
事务的隔离性和原子性分别通过锁和事务日志实现,而持久性则依赖于事务日志中的`Redo Log`。在MySQL中,`Redo Log`确保已提交事务的数据能持久保存,即使系统崩溃也能通过重做日志恢复数据。其工作原理是记录数据在内存中的更改,待事务提交时写入磁盘。此外,`Redo Log`采用简单的物理日志格式和高效的顺序IO,确保快速提交。通过不同的落盘策略,可在性能和安全性之间做出权衡。
1743 14
MySQL事务日志-Redo Log工作原理分析
|
2月前
|
存储 监控 安全
什么是日志管理,如何进行日志管理?
日志管理是对IT系统生成的日志数据进行收集、存储、分析和处理的实践,对维护系统健康、确保安全及获取运营智能至关重要。本文介绍了日志管理的基本概念、常见挑战、工具的主要功能及选择解决方案的方法,强调了定义管理目标、日志收集与分析、警报和报告、持续改进等关键步骤,以及如何应对数据量大、安全问题、警报疲劳等挑战,最终实现日志数据的有效管理和利用。
143 0
|
3月前
|
Python
log日志学习
【10月更文挑战第9天】 python处理log打印模块log的使用和介绍
49 0
|
3月前
|
数据可视化
Tensorboard可视化学习笔记(一):如何可视化通过网页查看log日志
关于如何使用TensorBoard进行数据可视化的教程,包括TensorBoard的安装、配置环境变量、将数据写入TensorBoard、启动TensorBoard以及如何通过网页查看日志文件。
297 0