.NET Framework之log4net组件的使用
一、前言
应用程序在生产环境运行中产生的错误信息、告警信息、特定调试信息或者接口对接信息如何让开发者进行查找?这时候可以使用日志来回溯系统执行的全过程查找系统问题,帮助开发者解决线上问题等。一般都会选择一个日志组件并且希望其是代码侵入少,使用便捷,写入性能高,比如开源的log4net。
二、使用
在项目引用中引入log4net组件,然后对日志配置文件进行配置,配置信息涉及到文件夹位置、文件名称格式、文件最大大小、文件的内容格式、日志级别等信息,比如Log4Net.config,通过这个日志文件定义基本信息,可以在打印日志生成想要的日志文件。
<?xml version="1.0" encoding="utf-8" ?> <configuration> <configSections> <section name="log4net" type="log4net.Config.Log4NetConfigurationSectionHandler, log4net"/> </configSections> <log4net> <appender name="ConsoleAppender" type="log4net.Appender.ConsoleAppender"> <layout type="log4net.Layout.PatternLayout"> <conversionPattern value="[%p %t] [%date{yyyy-MM-dd HH:mm:ss,fff}] %-5l - %m%n"/> </layout> </appender> <appender name="RollingLogFileAppender" type="log4net.Appender.RollingFileAppender"> <!--日志路径--> <param name= "File" value= "logs\Info\log_"/> <!--是否是向文件中追加日志--> <param name= "AppendToFile" value= "true"/> <!--log保留天数--> <param name= "MaxSizeRollBackups" value= "10"/> <!--日志文件名是否是固定不变的--> <param name= "StaticLogFileName" value= "false"/> <!--日志文件名格式为--> <param name= "DatePattern" value= "yyyy-MM-dd".log""/> <!--日志根据日期滚动--> <param name= "RollingStyle" value= "Date"/> <layout type="log4net.Layout.PatternLayout"> <conversionPattern value="记录时间:%date 线程ID:[%thread] 日志级别: %-5level 类:%logger - 描述:%message %n"/> </layout> <filter type="log4net.Filter.LevelRangeFilter"> <param name="LevelMin" value="Info" /> <param name="LevelMax" value="WARN" /> </filter> </appender> <appender name="RollingFile" type="log4net.Appender.RollingFileAppender"> <param name= "File" value= "logs\Error\Log_"/> <param name= "AppendToFile" value= "true"/> <param name= "MaxSizeRollBackups" value= "10"/> <param name= "StaticLogFileName" value= "false"/> <param name= "DatePattern" value= "yyyy-MM-dd".log""/> <param name= "RollingStyle" value= "Date"/> <layout type="log4net.Layout.PatternLayout"> <conversionPattern value="记录时间:%date 线程ID:[%thread] 日志级别: %-5level 类:%logger - 描述:%message %n"/> </layout> <filter type="log4net.Filter.LevelRangeFilter"> <param name="LevelMin" value="ERROR" /> <param name="LevelMax" value="FATAL" /> </filter> </appender> <root> <!--(高) OFF > FATAL > ERROR > WARN > INFO > DEBUG > ALL (低) --> <level value="ALL" /> <appender-ref ref="RollingLogFileAppender" /> <appender-ref ref="RollingFile" /> </root> </log4net> </configuration>
定义一个日志接口(主要是不同层级的写入方法),实现接口通过log4net组件来实现文件形式记录日志信息,使用组件读取配置文件、获取日志对象写入文件日志,还可以使用数据库方式实现接口写入日志。
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; namespace TQF.Logger.Logger { /// <summary> /// 定义日志接口类 /// </summary> public interface ILogger { /// <summary> /// 调试日志输出 /// </summary> /// <param name="msg">输出内容</param> void Debug(string msg); /// <summary> /// 调试日志输出 /// </summary> /// <param name="msg">输出内容</param> /// <param name="ex">输出异常</param> void Debug(string msg, Exception ex); /// <summary> /// 信息日志输出 /// </summary> /// <param name="msg">输出内容</param> void Info(string msg); /// <summary> /// 信息日志输出 /// </summary> /// <param name="msg">输出内容</param> /// <param name="ex">输出异常</param> void Info(string msg, Exception ex); /// <summary> /// 警告日志输出 /// </summary> /// <param name="msg">输出内容</param> void Warn(string msg);
/// <summary> /// 警告日志输出 /// </summary> /// <param name="msg">输出内容</param> /// <param name="ex">输出异常</param> void Warn(string msg, Exception ex); /// <summary> /// 错误日志输出 /// </summary> /// <param name="msg">输出内容</param> void Error(string msg); /// <summary> /// 错误日志输出 /// </summary> /// <param name="msg">输出内容</param> /// <param name="ex">输出异常</param> void Error(string msg, Exception ex); /// <summary> /// 致命日志输出 /// </summary> /// <param name="msg">输出内容</param> void Fatal(string msg); /// <summary> /// 致命日志输出 /// </summary> /// <param name="msg">输出内容</param> /// <param name="ex">输出异常</param> void Fatal(string msg, Exception ex); } }
using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Text; using System.Threading.Tasks; using log4net; using log4net.Appender; using log4net.Config; namespace TQF.Logger.Logger { /// <summary> /// 日志接口实现类 /// </summary> public class Log4Helper : ILogger { /// <summary> /// 考虑使用线程安全的字典 /// </summary> private Dictionary<string, ILog> LogDic = new Dictionary<string, ILog>(); /// <summary> /// 定义锁对象 /// </summary> private object _islock = new object(); /// <summary> /// 定义文件名 /// </summary> private string fileName = string.Empty;
/// <summary> /// 使用构造函数日志调用初始化 /// </summary> /// <param name="fileSavePath">日志文件保存路径[若路径为空,则默认程序根目录Logger文件夹;]</param> /// <param name="fileName">日志文件名[若文件名为空,则默认文件名:Default]</param> public Log4Helper(string fileSavePath, string fileName, string logSuffix = ".log") { try { Init(); if (string.IsNullOrEmpty(fileSavePath)) fileSavePath = "Logger"; if (string.IsNullOrEmpty(fileName)) fileName = "Default"; this.fileName = fileName; var repository = LogManager.GetRepository(); var appenders = repository.GetAppenders(); if (appenders.Length == 0) return; var targetApder = appenders.First(p => p.Name == "FileInfoAppender") as RollingFileAppender; targetApder.File = Path.Combine(fileSavePath, this.fileName + logSuffix); targetApder.ActivateOptions(); } catch (Exception ex) { } } /// <summary> /// 缓存日志对象(通过日志文件名缓存日志对象,减少日志对象的创建工作) /// </summary> /// <param name="name"></param> /// <returns></returns> private ILog GetLog(string name) { try { if (LogDic == null) { LogDic = new Dictionary<string, ILog>(); } lock (_islock) { if (!LogDic.ContainsKey(name)) { LogDic.Add(name, LogManager.GetLogger(name)); } } return LogDic[name]; } catch { return LogManager.GetLogger("Default"); } } /// <summary> /// 日志记录初始化 /// </summary> private void Init() { // 获取日志配置文件 var file = new FileInfo(Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Log4net.config")); // 检测配置文件是否存在 if (file.Exists) { //加载配置文件 XmlConfigurator.Configure(file); } else { new Exception("找不到日志配置文件!"); } } /// <summary> /// 调试日志输出 /// </summary> /// <param name="msg">内容</param> public void Debug(string msg) { var log = GetLog(this.fileName); if (log == null) { return; } log.Debug(msg); } /// <summary> /// 调试日志输出 /// </summary> /// <param name="msg">输出内容</param> /// <param name="ex">输出异常</param> public void Debug(string msg, Exception ex) { var log = GetLog(this.fileName); if (log == null) { return; } log.Debug(msg, ex); } /// <summary> /// 错误日志输出 /// </summary> /// <param name="msg">输出内容</param> public void Error(string msg) { var log = GetLog(this.fileName); if (log == null) { return; } log.Error(msg); } /// <summary> /// 错误日志输出 /// </summary> /// <param name="msg">输出内容</param> /// <param name="ex">输出异常</param> public void Error(string msg, Exception ex) { var log = GetLog(this.fileName); if (log == null) { return; } log.Error(msg, ex); } /// <summary> /// 致命日志输出 /// </summary> /// <param name="msg">输出内容</param> public void Fatal(string msg) { var log = GetLog(this.fileName); if (log == null) { return; } log.Fatal(msg); } /// <summary> /// 致命日志输出 /// </summary> /// <param name="msg">输出内容</param> /// <param name="ex">输出异常</param> public void Fatal(string msg, Exception ex) { var log = GetLog(this.fileName); if (log == null) { return; } log.Fatal(msg, ex); } /// <summary> /// 信息日志输出 /// </summary> /// <param name="msg">输出内容</param> public void Info(string msg) { var log = GetLog(this.fileName); if (log == null) { return; } log.Info(msg); } /// <summary> /// 信息日志输出 /// </summary> /// <param name="msg">输出内容</param> /// <param name="ex">输出异常</param> public void Info(string msg, Exception ex) { var log = GetLog(this.fileName); if (log == null) { return; } log.Info(msg, ex); } /// <summary> /// 警告日志输出 /// </summary> /// <param name="msg">输出内容</param> public void Warn(string msg) { var log = GetLog(this.fileName); if (log == null) { return; } log.Warn(msg); } /// <summary> /// 警告日志输出 /// </summary> /// <param name="msg">输出内容</param> /// <param name="ex">输出异常</param> public void Warn(string msg, Exception ex) { var log = GetLog(this.fileName); if (log == null) { return; } log.Warn(msg, ex); } } }
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using TQF.Logger.Logger; namespace TQF.Logger { /// <summary> /// 配置文件如果不设置属性“复制到输出目录”不会在debug文件中存在 /// </summary> class Program { static void Main(string[] args) { var log4Helper = new Log4Helper(string.Empty, "program"); log4Helper.Info("info"); } } }
按照配置文件格式定义的文件夹、文件名称、文件内容格式,相对标准化,但是要特殊的记录日志信息比如不确定文件夹名称、不确定的文件名称等,就要开发者自定义,通过在写入日志的时候,传入文件夹和文件名称形式来存储,对于特殊的日志信息可以通过该方法达到特定要求,如下所示。
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Collections.Concurrent; using System.Configuration; using log4net; using log4net.Appender; using log4net.Core; using log4net.Layout; using log4net.Repository; using log4net.Repository.Hierarchy; [assembly: log4net.Config.XmlConfigurator(Watch = true)] namespace DataPullAPI.Utility { /// <summary> /// log4net 自定义日志类 /// </summary> public static class LogCustomHelper { /// <summary> /// 类型安全的字典对象 /// </summary> private static readonly ConcurrentDictionary<string, ILog> loggerContainer = new ConcurrentDictionary<string, ILog>(); //默认配置 private const int MAX_SIZE_ROLL_BACKUPS = 20; private const string LAYOUT_PATTERN = "%message"; private const string DATE_PATTERN = "yyyyMMdd"; private const string MAXIMUM_FILE_SIZE = "10MB"; private const string LEVEL = "ALL"; /// <summary> /// 获取日志对象 /// </summary> /// <param name="loggerName"></param> /// <param name="category"></param> /// <param name="additivity"></param> /// <returns></returns> public static ILog GetCustomLogger(string loggerName, string category = null, bool additivity = false) { return loggerContainer.GetOrAdd(loggerName, delegate (string name) { RollingFileAppender newAppender = GetNewFileApender(loggerName, GetFile(category, loggerName), MAX_SIZE_ROLL_BACKUPS, true, true, MAXIMUM_FILE_SIZE, RollingFileAppender.RollingMode.Composite, DATE_PATTERN, LAYOUT_PATTERN); Hierarchy repository = (Hierarchy)LogManager.GetRepository(); Logger logger = repository.LoggerFactory.CreateLogger(repository, loggerName); logger.Hierarchy = repository; logger.Parent = repository.Root; logger.Level = GetLoggerLevel(LEVEL); logger.Additivity = additivity; logger.AddAppender(newAppender); logger.Repository.Configured = true; return new LogImpl(logger); }); } /// <summary> /// 自定日志文件夹和文件路径 /// </summary> /// <param name="category"></param> /// <param name="loggerName"></param> /// <returns></returns> private static string GetFile(string category, string loggerName) { if (string.IsNullOrEmpty(category)) { return string.Format(@"Logs\{0}\{1}.txt", DateTime.Now.ToString("yyyy-MM-dd"), loggerName); } else { return string.Format(@"Logs\{0}\{1}\{2}.txt", category,DateTime.Now.ToString("yyyy-MM-dd"), loggerName); } } /// <summary> /// 获取日志级别 /// </summary> /// <param name="level"></param> /// <returns></returns> private static Level GetLoggerLevel(string level) { if (!string.IsNullOrEmpty(level)) { switch (level.ToLower().Trim()) { case "debug": return Level.Debug; case "info": return Level.Info; case "warn": return Level.Warn; case "error": return Level.Error; case "fatal": return Level.Fatal; } } return Level.Debug; } /// <summary> /// 获取配置信息 /// </summary> /// <param name="appenderName"></param> /// <param name="file"></param> /// <param name="maxSizeRollBackups"></param> /// <param name="appendToFile"></param> /// <param name="staticLogFileName"></param> /// <param name="maximumFileSize"></param> /// <param name="rollingMode"></param> /// <param name="datePattern"></param> /// <param name="layoutPattern"></param> /// <returns></returns> private static RollingFileAppender GetNewFileApender(string appenderName, string file, int maxSizeRollBackups, bool appendToFile = true, bool staticLogFileName = false, string maximumFileSize = "2MB", RollingFileAppender.RollingMode rollingMode = RollingFileAppender.RollingMode.Size, string datePattern = "yyyyMMdd\".txt\"", string layoutPattern = "%d [%t] %-5p %c - %m%n") { RollingFileAppender appender = new RollingFileAppender { LockingModel = new FileAppender.MinimalLock(), Name = appenderName, File = file, AppendToFile = appendToFile, MaxSizeRollBackups = maxSizeRollBackups, MaximumFileSize = maximumFileSize, StaticLogFileName = staticLogFileName, RollingStyle = rollingMode, DatePattern = datePattern }; PatternLayout layout = new PatternLayout(layoutPattern); appender.Layout = layout; layout.ActivateOptions(); appender.ActivateOptions(); return appender; } /// <summary> /// 写入日志信息 /// </summary> /// <param name="folderName">文件夹名称</param> /// <param name="fileName">文件名称</param> /// <param name="message">日志信息</param> public static void Info(string folderName, string fileName,string message) { ILog logger = GetCustomLogger(fileName, folderName); logger.Info(message); } } }
三、总结
在生产环境记录日志可以帮助用户定位问题,解决问题,对系统执行过程中可能出现异常错误进行捕获,对执行全过程进行回溯查看等;在存储方式上不仅可以使用文本文件方式保存,还可以使用数据库保存(关系型数据,非关系型数据库mongodb);在分布式系统中日志的聚集,整合,可视化,统一管理的考量与实践。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?