Log4Net日誌分類和自動維護

来源:http://www.cnblogs.com/shouce/archive/2016/01/07/5108051.html
-Advertisement-
Play Games

背景 在程式中,我們調試運行時信息,Log4Net是一個不錯的解決方案。不知道是我用的不好,用到最後反而都不想看日誌了。原因是因為我n個功能使用的預設的Logger來記錄日誌,這樣以來,所有功能記錄的信息都依賴同一個配置,於是所有的信息都在一個文件中,有時候查找起來,極其不方便。 我想,能不能按.....


背景

  在程式中,我們調試運行時信息,Log4Net是一個不錯的解決方案。不知道是我用的不好,用到最後反而都不想看日誌了。原因是因為我n個功能使用的預設的Logger來記錄日誌,這樣以來,所有功能記錄的信息都依賴同一個配置,於是所有的信息都在一個文件中,有時候查找起來,極其不方便。

  我想,能不能按照功能分類呢?如果通過配置不同的logger,然後功能根據不同的LoggerName載入Ilog實例,是可以做到。但由於這些功能的log配置差異性極小,也許僅僅就是文件名不同。於是想通過代碼進行配置。

解決方案

  想來這樣的需求應該也算正常,但經過查找,發現在log4j中有相關的咨詢,但看不懂,只能自己下載log4net的源碼,看看它的運行原理。

  如果我們需要通過配置不同的ILog,流程是這樣的,首先要創建一個ILoggerRepository,通過它來進行日誌等級的配置,和各種Appender,接著通過LogManager在ILoggerRepository獲取一個Ilog對象,就可以寫日誌了。代碼如下:

複製代碼
        public static ILog GetLogger(string repositoryName = "")
        {
            if (string.IsNullOrEmpty(repositoryName)) return LogManager.GetLogger("Defalut");

            ILoggerRepository repository = null;
            try
            {
                repository = LogManager.GetRepository(repositoryName);
            }
            catch (Exception) { }

            //找到直接返回ilog
            if (repository != null)
                return LogManager.GetLogger(repositoryName, "Defalut");

            //未找到則創建,多線程下很有可能創建時,就存在了
            try
            {
                repository = LogManager.CreateRepository(repositoryName);
            }
            catch (Exception)
            {
                repository = LogManager.GetRepository(repositoryName);
            }

            //配置日誌等級  讀取Appsettings預設配置  
            var appSet = ConfigurationManager.AppSettings.AllKeys;

            //查找日誌等級
            const string logLevel = "LogLevel";
            var hasSettings = Array.IndexOf(appSet, logLevel);
            if (hasSettings > -1)
            {
                var level = ConfigurationManager.AppSettings[logLevel].ToLower();
                if (level == "all") repository.Threshold = Level.All;
                else if (level == "debug") repository.Threshold = Level.Debug;
                else if (level == "info") repository.Threshold = Level.Info;
                else if (level == "warn") repository.Threshold = Level.Warn;
                else if (level == "error") repository.Threshold = Level.Error;
                else if (level == "fatal") repository.Threshold = Level.Fatal;
                else if (level == "off") repository.Threshold = Level.Off;
            }
            else repository.Threshold = Level.All;

            //查找輸出Appender
            const string logAppender = "LogAppender";
            hasSettings = Array.IndexOf(appSet, logAppender);
            if (hasSettings > -1)
            {
                var appenders = ConfigurationManager.AppSettings[logAppender].ToLower().Split(',');
                foreach (var appender in appenders)
                {
                    if (appender == "rollingfile") LoadRollingFileAppender(repository);
                    else if (appender == "console") LoadConsoleAppender(repository);
                    else if(appender == "trace") LoadTraceAppender(repository);
                }
            }
            else LoadRollingFileAppender(repository);

            return LogManager.GetLogger(repositoryName, "Default");
        }
複製代碼

  Log4net 會自動維護ILog和ILoggerRepository,所以不用保存,通過LogManger獲取即可。但是我沒有找到方法來查找指定的ILoggerRepository是否存在,所以在上述代碼中通過try catch進行判斷。由於我們是通過代碼方式配置log4net,但仍舊希望通過配置來設置日誌等級和輸出類型,所以我選擇在Appsettings文件中進行配置,一次配置,針對所有的ILoggerRepository都生效。之所以不是針對每一個功能進行配置,主要是由於配置麻煩,而且如果真要這麼做,直接在log4net中配置不同的logger,通過loggername引用,可能會更好些。

日誌維護

  於是接下來的問題是,如何刪除日誌?

  log4net可以配置RollingFileAppender,翻譯過來是滾動文件。通過設置MaximumFileSize和MaxSizeRollBackups 2個參數可以完成這樣的目的,該日誌文件每達到一定大小就會產生一個新文件,文件最大總數則為MaxSizeRollBackups,但前提的是文件名一致。

  比如我們採用如下配置:

複製代碼
    <!--RollingFileAppender:輸出到文件中-->
    <appender name="SysAppender" type="log4net.Appender.RollingFileAppender">
      <!--日誌的路徑-->
      <file value="Logs/" />
      <!--是否覆蓋,預設是追加true-->
      <appendToFile value="true"/>
      <!--不占用日誌文件進程-->
      <lockingModel type="log4net.Appender.FileAppender+MinimalLock" />
      <rollingStyle value="Composite"/>
      <!--文件名稱-->
      <DatePattern value="yyyy-MM-dd HH'時.log'"></DatePattern>
      <!--設置無限備份=-1 ,最大備份數為1000-->
      <param name="MaxSizeRollBackups" value="1000"/>
      <!--每個文件的大小-->
      <param name="MaximumFileSize" value="500KB"/>
      <!--名稱是否可以更改 為false為可以更改-->
      <param name="StaticLogFileName" value="false"/>
      <layout type="log4net.Layout.PatternLayout">
        <!--輸出格式-->
        <conversionPattern value="%n【記錄時間】%date%n【描述】%message%n"/>
      </layout>
    </appender>
複製代碼

  效果則是 日誌以小時為單位,每小時最大備份總數為1000,每到500kb後形成日誌文件。當隨著天數的增加,文件也開始變多,需要我們自己去清理。所以上述配置可以稍微改下:

複製代碼
    <!--RollingFileAppender:輸出到文件中-->
    <appender name="SysAppender" type="log4net.Appender.RollingFileAppender">
      <!--日誌的路徑-->
      <file value="Logs/日誌.log" />
      <!--是否覆蓋,預設是追加true-->
      <appendToFile value="true"/>
      <!--不占用日誌文件進程-->
      <lockingModel type="log4net.Appender.FileAppender+MinimalLock" />
      <rollingStyle value="Composite"/>
      <!--文件名稱-->
      <DatePattern value="yyyy-MM-dd HH'時.log'"></DatePattern>
      <!--設置無限備份=-1 ,最大備份數為1000-->
      <param name="MaxSizeRollBackups" value="1000"/>
      <!--每個文件的大小-->
      <param name="MaximumFileSize" value="500KB"/>
      <!--名稱是否可以更改 為false為可以更改-->
      <param name="StaticLogFileName" value="true"/>
      <layout type="log4net.Layout.PatternLayout">
        <!--輸出格式-->
        <conversionPattern value="%n【記錄時間】%date%n【描述】%message%n"/>
      </layout>
    </appender>
複製代碼

  主要是設置了日誌文件名,而且是靜態名稱,不允許更改。這樣以來,日誌文件最多只有1000個了,就不用手工維護了,你會看到這樣的日誌文件名:

  日誌.log   日誌.log.1  日誌.log.2   ... 預設是文件名最後的數字越大,時間最老,所以查看新日誌,主要是在 第一個文件名 中。

  另外提醒下:如果設置Appender使用了靜態文件名,那麼歷史日誌會被覆蓋,而極其在意歷史日誌的應用,建議不要使用此方案,清理日誌的話,還是老老實實寫代碼吧。

測試代碼下載


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 不廢話,直接上代碼,您也可以點這裡看瀑布流效果演示。前臺:getMyVal('category',$_GET);$xiaohuaList=Xiaohua::model()->getXiaohao($category); //打開頁面預設顯示的數據?> id;?> ">title);?> ...
  • 最近在學python,正好遇到學校需要選宿舍,就用python寫了一個搶宿舍的軟體。其中有一個模塊是用來登陸的,登陸的時候需要輸入驗證碼,不過後來發現了直接可以繞過驗證碼直接登陸的bug。不過這是另外的話題,開始的時候我並沒有發現這個隱藏起來的秘密,所以我就寫了這個python代碼段用來實現解析驗證...
  • //#include#include#includestruct Word //定義結構數組儲存單詞{ char w[20]; int num;}Word[10000];void main() ...
  • class FileUtils{ //文件目錄下文件總數目 public static int fileNumber(File dir) { int filenumber = 0; if(dir.exists()) { ...
  • 在 FMX 下的 TComboBox 下拉菜單字型修改有二種方法:使用 Style,需先設定好 Style 後,再指定預設項的 Style,方法如下:procedure TForm1.FormCreate(Sender: TObject);var i: Integer;begin Combo...
  • String和StringBuffer、StringBuilder的區別
  • 第一節 介面慨述 介面(interface)用來定義一種程式的協定。實現介面的類或者結構要與介面的定義嚴格一致。有了這個協定,就可以拋開編程語言的限制(理論上)。介面可以從多個基介面繼承,而類或結構可以實現多個介面。介面可以包含方法、屬性、事件和索引器。介面本身不提供它所定義的成員的實現。介面只指....
  • protected void btnSend_Click(object sender, EventArgs e) 2: { 3: int temp = -1; 4: int exhibitionID = int.Parse(Request.QueryString["Con...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...