ASP.NET WebApi項目框架搭建(四):日志管理之log4net


一、前言

在項目開發的過程中,日志是必不可少的一部分,可以幫我們定位異常,分析錯誤原因等功能,log4net是.Net下一個非常優秀的開源日志記錄組件。log4net記錄日志的功能非常強大。它可以將日志分不同的等級,以不同的格式,輸出到不同的媒介。本文主要是介紹在前幾篇已經創建好的項目里加上日志處理機制

二、安裝並使用log4net

1.使用nuget安裝log4net

 

2.log4net的用法類似下面的代碼:

        [Route("log4")]
        [HttpGet]
        public IHttpActionResult Log()
        {
            // 通過LogManager的靜態方法GetLogger生成一個Ilog對象
            ILog log = LogManager.GetLogger(typeof(IndexController)); // 下面是日志處理
            log.Debug("測試debug", new Exception("debug異常"));
            log.Info("測試Info", new Exception("Info異常"));
            log.Warn("測試Warn", new Exception("Warn異常"));
            log.Error("測試Error", new Exception("Error異常"));
            log.Fatal("測試Fatal", new Exception("Fatal異常"));
            return Ok("已經寫入日志");

        }

3.運行項目,瀏覽器調用請求

三、通過autofac解耦Ilog對象

1.我們用autofac的Module方式去注冊log4net組件,項目AutoFac目錄下新建Module文件夾,並新建LoggingModule類

 public class LoggingModule : Autofac.Module
    {
        private static void InjectLoggerProperties(object instance)
        {
            var instanceType = instance.GetType();

            // Get all the injectable properties to set.
            // If you wanted to ensure the properties were only UNSET properties,
            // here's where you'd do it.
            var properties = instanceType
              .GetProperties(BindingFlags.Public | BindingFlags.Instance)
              .Where(p => p.PropertyType == typeof(ILog) && p.CanWrite && p.GetIndexParameters().Length == 0);

            // Set the properties located.
            foreach (var propToSet in properties)
            {
                propToSet.SetValue(instance, LogManager.GetLogger(instanceType), null);
            }
        }

        private static void OnComponentPreparing(object sender, PreparingEventArgs e)
        {
            e.Parameters = e.Parameters.Union(
              new[]
              {
        new ResolvedParameter(
            (p, i) => p.ParameterType == typeof(ILog),
            (p, i) => LogManager.GetLogger(p.Member.DeclaringType)
        ),
              });
        }

        protected override void AttachToComponentRegistration(IComponentRegistry componentRegistry, IComponentRegistration registration)
        {
            registration.Preparing += OnComponentPreparing;

            // Handle properties.
            registration.Activated += (sender, e) => InjectLoggerProperties(e.Instance);
        }
    }

這段代碼為autofac官網里提供的,參考地址:http://autofaccn.readthedocs.io/en/latest/examples/log4net.html?highlight=module

 2.ContainerBuilerCommon下注冊module

 public static IContainer GetWebApiContainer()
        {
            var builder = new ContainerBuilder();
            // 注冊webapi的所有控制器
            builder.RegisterApiControllers(Assembly.GetExecutingAssembly());
            // 注冊一個用於測試的組件。
            builder.RegisterType<Person>();
            //注冊日志組件
            builder.RegisterModule<LoggingModule>();
            return builder.Build();
        }

3.控制器通過構造函數注入Ilog

        private ILog _log;
        public IndexController(ILog log, Person person)
        {
            _person = person;
            _log = log;
        }

4.新建get請求

        [Route("autolog4")]
        [HttpGet]
        public IHttpActionResult AutoLog()
        {
            // 通過LogManager的靜態方法GetLogger生成一個Ilog對象
            _log.Debug("測試debug", new Exception("debug異常"));
            _log.Info("測試Info", new Exception("Info異常"));
            _log.Warn("測試Warn", new Exception("Warn異常"));
            _log.Error("測試Error", new Exception("Error異常"));
            _log.Fatal("測試Fatal", new Exception("Fatal異常"));
            return Ok("已經寫入日志");

        }

5.現在編譯程序后用postman工具的get方法訪問接口:http://localhost:xxx/api/autolog4,程序是運行正常的 。但現在日志即沒有寫入到某個文件、數據庫或是發送到郵件里,也沒有輸出到控制台上。這就是log4net的設計的好處,在程序里你只管做日志的處理,如調用ILog的Debug()、Info()、Warn()、Error()、Fatal(),至於日志是由什么機制去處理(如寫入文件,寫入數據庫等)是由另一個流程來控制,即log4net的配置文件。如果程序里沒有log4net的配置文件,程序也能正常運行。

四、配置Log4net

1.新建Log4net.config,代碼如下

<?xml version="1.0" encoding="utf-8"?>
<configuration>
    <configSections>
        <section name="log4net" type="log4net.Config.Log4NetConfigurationSectionHandler, log4net"/>
    </configSections>

    <system.web>
        <compilation debug="true" targetFramework="4.5.2" />
        <httpRuntime targetFramework="4.5.2" />
    </system.web>
<log4net>
    <!--錯誤日志:::記錄錯誤日志-->
    <!--按日期分割日志文件 一天一個-->
    <!-- appender 定義日志輸出方式   將日志以回滾文件的形式寫到文件中。-->
    <appender name="ErrorAppender" type="log4net.Appender.RollingFileAppender">
        <!--保存路徑:下面路徑項目啟動的時候自動在C盤中創建log、logError文件-->
        <file value="log/error/error_" />
        <!-- 如果想在本項目中添加路徑,那就直接去掉C:\\  只設置log\\LogError   項目啟動中默認創建文件 -->
        <appendToFile value="true"/>
        <!--按照何種方式產生多個日志文件(日期[Date],文件大小[Size],混合[Composite])-->
        <rollingStyle value="Date"/>
        <!--這是按日期產生文件夾-->
        <datePattern value="yyyy-MM-dd'.log'"/>
        <!--是否只寫到一個文件中-->
        <staticLogFileName value="false"/>
        <!--保留的log文件數量 超過此數量后 自動刪除之前的   好像只有在 按Size分割時有效 設定值value="-1"為不限文件數-->
        <param name="MaxSizeRollBackups" value="100"/>
        <!--每個文件的大小。只在混合方式與文件大小方式下使用。超出大小后在所有文件名后自動增加正整數重新命名,數字最大的最早寫入。可用的單位:KB|MB|GB。不要使用小數,否則會一直寫入當前日志-->
        <maximumFileSize value="50M" />
        <!-- layout 控制Appender的輸出格式,也可以是xml  一個Appender只能是一個layout-->
        <layout type="log4net.Layout.PatternLayout">
            <!--每條日志末尾的文字說明-->
            <!--輸出格式 模板-->
            <!-- <param name="ConversionPattern"  value="記錄時間:%date 線程ID:[%thread] 日志級別:%-5level 記錄類:%logger   
        操作者ID:%property{Operator} 操作類型:%property{Action}%n  當前機器名:%property%n當前機器名及登錄用戶:%username %n  
        記錄位置:%location%n 消息描述:%property{Message}%n   異常:%exception%n 消息:%message%newline%n%n" />-->

            <!--樣例:2008-03-26 13:42:32,111 [10] INFO  Log4NetDemo.MainClass [(null)] - info-->
            <!--<conversionPattern value="%newline %n記錄時間:%date %n線程ID:[%thread] %n日志級別: %-5level %n錯誤描述:%message%newline %n"/>-->
            <conversionPattern value="%n==========
                                  %n【日志級別】%-5level
                                  %n【記錄時間】%date
                                  %n【執行時間】[%r]毫秒
                                  %n【出錯文件】%F
                                  %n【出錯行號】%L
                                  %n【出錯的類】%logger 屬性[%property{NDC}]
                                  %n【錯誤描述】%message
                                  %n【錯誤詳情】%newline"/>
        </layout>
        <filter type="log4net.Filter.LevelRangeFilter,log4net">
            <levelMin value="ERROR" />
            <levelMax value="FATAL" />
        </filter>
    </appender>

    <!--DEBUG:::記錄DEBUG日志-->
    <!--按日期分割日志文件 一天一個-->
    <!-- appender 定義日志輸出方式   將日志以回滾文件的形式寫到文件中。-->
    <appender name="DebugAppender" type="log4net.Appender.RollingFileAppender">
        <!--保存路徑:下面路徑項目啟動的時候自動在C盤中創建log、logError文件-->
        <file value="log/debug/debug_" />
        <!-- 如果想在本項目中添加路徑,那就直接去掉C:\\  只設置log\\LogError   項目啟動中默認創建文件 -->
        <appendToFile value="true"/>
        <!--按照何種方式產生多個日志文件(日期[Date],文件大小[Size],混合[Composite])-->
        <rollingStyle value="Date"/>
        <!--這是按日期產生文件夾-->
        <datePattern value="yyyy-MM-dd'.log'"/>
        <!--是否只寫到一個文件中-->
        <staticLogFileName value="false"/>
        <!--保留的log文件數量 超過此數量后 自動刪除之前的   好像只有在 按Size分割時有效 設定值value="-1"為不限文件數-->
        <param name="MaxSizeRollBackups" value="100"/>
        <!--每個文件的大小。只在混合方式與文件大小方式下使用。超出大小后在所有文件名后自動增加正整數重新命名,數字最大的最早寫入。可用的單位:KB|MB|GB。不要使用小數,否則會一直寫入當前日志-->
        <maximumFileSize value="50M" />
        <!-- layout 控制Appender的輸出格式,也可以是xml  一個Appender只能是一個layout-->
        <layout type="log4net.Layout.PatternLayout">
            <!--每條日志末尾的文字說明-->
            <!--輸出格式 模板-->
            <!-- <param name="ConversionPattern"  value="記錄時間:%date 線程ID:[%thread] 日志級別:%-5level 記錄類:%logger   
        操作者ID:%property{Operator} 操作類型:%property{Action}%n  當前機器名:%property%n當前機器名及登錄用戶:%username %n  
        記錄位置:%location%n 消息描述:%property{Message}%n   異常:%exception%n 消息:%message%newline%n%n" />-->

            <!--樣例:2008-03-26 13:42:32,111 [10] INFO  Log4NetDemo.MainClass [(null)] - info-->
            <!--<conversionPattern value="%newline %n記錄時間:%date %n線程ID:[%thread] %n日志級別: %-5level %n錯誤描述:%message%newline %n"/>-->
            <conversionPattern value="%n==========
                                  %n【日志級別】%-2level
                                  %n【記錄時間】%date
                                  %n【執行時間】[%r]毫秒
                                  %n【debug文件】%F
                                  %n【debug行號】%L
                                  %n【debug類】%logger 屬性[%property{NDC}]
                                  %n【debug描述】%message"/>
        </layout>
        <filter type="log4net.Filter.LevelRangeFilter,log4net">
            <levelMin value="DEBUG" />
            <levelMax value="WARN" />
        </filter>
    </appender>
    <!--Set root logger level to DEBUG and its only appender to A1-->
    <root>
        <!--控制級別,由低到高: ALL|DEBUG|INFO|WARN|ERROR|FATAL|OFF-->
        <level value="ALL" />
        <appender-ref ref="DebugAppender" />
        <appender-ref ref="ErrorAppender" />
    </root>
</log4net>
</configuration>

2.properties下的AssemblyInfo.cs文件里添加以下代碼:

[assembly: XmlConfigurator(Watch = true, ConfigFile = "Log4Net.config")]

3.運行程序,調用日志請求,log4net會在項目主目錄下的log文件夾生成日志文件

 

 

 

DEBUG日志:

 

ERROR日志:

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM