


最近在使用log4net,在使用之前我们必须知道文件流是如何操作的,否则就是盲人摸向。。。,在FileAppender.cs文件里面有LockingModelBase来控制流的锁,默认有3个子类
ExclusiveLock:默认的,Hold an exclusive lock on the output file,Open the file once for writing and hold it open until CloseFile is called. Maintains an exclusive lock on the file during this time.
MinimalLock:Acquires the file lock for each write,Opens the file once for each AcquireLock / ReleaseLock cycle, thus holding the lock for the minimal amount of time.This method of locking is considerably slower than FileAppender.ExclusiveLock but allows other processes to move/delete the log file whilst logging continues.
InterProcessLock:Provides cross-process file locking.使用Mutex来实现多进程
这里意思是MinimalLock比ExclusiveLock慢一点,因为它每次都会打开关闭文件流。
不过有2个类感觉比较重要PatternString.cs
和PatternLayout.cs
如果log文件在一个公共的目录,建议大家log文件加上计算机名称、应用程序名称、进程ID(如web 有多个工作者) 如:
<file type="log4net.Util.PatternString" value="\\192.168.0.1\logs\%env{COMPUTERNAME}\%appsetting{ApplicationName}\%processid\Log\" />
但是这里的log记录默认都是采用同步方式的,但是我个人更趋向用异步多线程的思路来写log,首先log的信息记录在内存ConcurrentQueue里面,然后在通过一个后台线程把ConcurrentQueue里面的东西记录到文件流里面。至于性能高出多少我想就不用多说了吧,写内存肯定比写流快啊
具体实现code如下:
[assembly: log4net.Config.XmlConfigurator(Watch = true, ConfigFile = "log4net.config")]namespace ConsoleApp { using log4net; using System; using System.Collections.Concurrent; using System.Threading; using System.Threading.Tasks; public sealed class QueueLogger { /// <summary> /// 记录消息Queue /// </summary> private readonly ConcurrentQueue<QueueLogMessage> _que; /// <summary> /// 信号 /// </summary> private readonly ManualResetEvent _mre; /// <summary> /// 日志 /// </summary> private readonly ILog _log; /// <summary> /// 日志 /// </summary> private static QueueLogger flashLog = new QueueLogger(); private QueueLogger() { // 设置日志配置文件路径 //XmlConfigurator.Configure(new FileInfo(Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "log4net.config"))); _que = new ConcurrentQueue<QueueLogMessage>(); _mre = new ManualResetEvent(false); _log = LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType); Task.Run(() => { WriteLog(); }); } /// <summary> /// 从队列中写日志至磁盘 /// </summary> private void WriteLog() { while (true) { // 等待信号通知 _mre.WaitOne(); QueueLogMessage msg; // 判断是否有内容需要如磁盘 从列队中获取内容,并删除列队中的内容 while (_que.Count > 0 && _que.TryDequeue(out msg)) { // 判断日志等级,然后写日志 switch (msg.Level) { case QueueLogLevel.Debug: _log.Debug(msg.Message, msg.Exception); break; case QueueLogLevel.Info: _log.Info(msg.Message, msg.Exception); break; case QueueLogLevel.Error: _log.Error(msg.Message, msg.Exception); break; case QueueLogLevel.Warn: _log.Warn(msg.Message, msg.Exception); break; case QueueLogLevel.Fatal: _log.Fatal(msg.Message, msg.Exception); break; } } // 重新设置信号 _mre.Reset(); } } /// <summary> /// 写日志 /// </summary> /// <param name="message">日志文本</param> /// <param name="level">等级</param> /// <param name="ex">Exception</param> public void EnqueueMessage(string message, QueueLogLevel level, Exception ex = null) { if ((level == QueueLogLevel.Debug && _log.IsDebugEnabled) || (level == QueueLogLevel.Error && _log.IsErrorEnabled) || (level == QueueLogLevel.Fatal && _log.IsFatalEnabled) || (level == QueueLogLevel.Info && _log.IsInfoEnabled) || (level == QueueLogLevel.Warn && _log.IsWarnEnabled)) { _que.Enqueue(new QueueLogMessage { // Message = "[" + DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss,fff") + "]\r\n" + message, Message = message, Level = level, Exception = ex }); // 通知线程往磁盘中写日志 _mre.Set(); } } public static void Debug(string msg, Exception ex = null) { flashLog.EnqueueMessage(msg, QueueLogLevel.Debug, ex); } public static void Error(string msg, Exception ex = null) { flashLog.EnqueueMessage(msg, QueueLogLevel.Error, ex); } public static void Fatal(string msg, Exception ex = null) { flashLog.EnqueueMessage(msg, QueueLogLevel.Fatal, ex); } public static void Info(string msg, Exception ex = null) { flashLog.EnqueueMessage(msg, QueueLogLevel.Info, ex); } public static void Warn(string msg, Exception ex = null) { flashLog.EnqueueMessage(msg, QueueLogLevel.Warn, ex); } } /// <summary> /// 日志等级 /// </summary> public enum QueueLogLevel { Debug, Info, Error, Warn, Fatal } /// <summary> /// 日志内容 /// </summary> public class QueueLogMessage { public string Message { get; set; } public QueueLogLevel Level { get; set; } public Exception Exception { get; set; } } }
至于CSV格式有2中方法 实现,一是自定义PatternLayout类:
namespace log4net { using Layout; using System.IO; using System.Text; using Util; using Core; public class CSVPatternLayout : PatternLayout { public override void ActivateOptions() { AddConverter("newfield", typeof(CSVNewFiledConverter)); AddConverter("endrow", typeof(CSVEndRowConverter)); base.ActivateOptions(); } public override void Format(TextWriter writer, LoggingEvent loggingEvent) { var csvWriter = new CSVTextWriter(writer); csvWriter.WriteQuote(); base.Format(csvWriter, loggingEvent); } } public class CSVTextWriter : TextWriter { private readonly TextWriter textWriter; public CSVTextWriter(TextWriter txtWriter) { textWriter = txtWriter; } public override void Write(char value) { // base.Write(value); textWriter.Write(value); //if (value == '"') //{ //} } public void WriteQuote() { textWriter.Write('"'); } public override Encoding Encoding { get { return textWriter.Encoding; } } } public class CSVNewFiledConverter : PatternConverter { protected override void Convert(TextWriter writer, object state) { var csvWriter = writer as CSVTextWriter; csvWriter?.WriteQuote(); writer.Write(","); csvWriter?.WriteQuote(); } } public class CSVEndRowConverter : PatternConverter { protected override void Convert(TextWriter writer, object state) { var csvWriter = writer as CSVTextWriter; csvWriter?.WriteQuote(); writer.WriteLine(); } } }
配置文件中需要加上逗号
<layout type="log4net.CSVPatternLayout,ConsoleApp"> <header value="Time,Thread,Level,Logger,Message,Exception " /> <conversionPattern value="%date{yyyy-MM-dd HH:mm:ss} %newfield %thread%newfield%level%newfield%logger%newfield%message%newfield%exception %endrow " /> </layout>
这里 是\r\n,%newfield是一个逗号,%endrow是逗号+换行
看到这里其实我们可以自己拼接CSV的内容,也就是说只要有,\r\n就可以了
<layout type="log4net.Layout.PatternLayout"> <header value="Time,Message,Type , " /> <param name="ConversionPattern" value=" "%date{yyyy-MM-dd HH:mm:ss}","%message%" "/> </layout>
调用code:
StringBuilder sb = new StringBuilder(); sb.Append("test"); sb.Append("\",\""); sb.Append("debug"); QueueLogger.Debug(sb.ToString());
写入的信息是test","debug,在加上ConversionPattern里面的配置就是"test","debug".整个配置如下:
<?xml version="1.0" encoding="utf-8"?> <configuration> <configSections> <section name="log4net" type="log4net.Config.Log4NetConfigurationSectionHandler, log4net" /> </configSections> <log4net xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <appender name="InfoLog" type="log4net.Appender.RollingFileAppender"> <param name="File" value="Log\Info\Info" /> <param name="AppendToFile" value="True" /> <appendToFile value="true" /> <maxSizeRollBackups value="100" /> <maximumFileSize value="10MB" /> <staticLogFileName value="false" /> <rollingStyle value="Composite" /> <datePattern value="yyyyMMdd'.csv'" /> <lockingModel type="log4net.Appender.FileAppender+MinimalLock" /> <layout type="log4net.CSVPatternLayout,ConsoleApp"> <header value="Time,Thread,Level,Logger,Message,Exception " /> <conversionPattern value="%date{yyyy-MM-dd HH:mm:ss}%newfield%thread%newfield%level%newfield%logger%newfield%message%newfield%exception%endrow" /> </layout> <filter type="log4net.Filter.LevelRangeFilter"> <param name="LevelMin" value="INFO" /> <param name="LevelMax" value="INFO" /> </filter> </appender> <appender name="DebugLog" type="log4net.Appender.RollingFileAppender"> <file type="log4net.Util.PatternString" value="Log\Debug\Debug" /> <appendToFile value="true" /> <maxSizeRollBackups value="100" /> <maximumFileSize value="10MB" /> <staticLogFileName value="false" /> <rollingStyle value="Composite" /> <datePattern value="yyyyMMdd'.csv'" /> <lockingModel type="log4net.Appender.FileAppender+ExclusiveLock" /> <layout type="log4net.Layout.PatternLayout"> <header value="Time,Message,Type, " /> <param name="ConversionPattern" value=""%date{yyyy-MM-dd HH:mm:ss}","%message%" "/> </layout> <filter type="log4net.Filter.LevelRangeFilter"> <param name="LevelMin" value="DEBUG" /> <param name="LevelMax" value="DEBUG" /> </filter> </appender> <appender name="ColoredConsoleAppender" type="log4net.Appender.ColoredConsoleAppender"> <mapping> <level value="ERROR" /> <foreColor value="Red" /> </mapping> <mapping> <level value="INFO" /> <foreColor value="Green" /> </mapping> <layout type="log4net.Layout.PatternLayout"> <conversionPattern value="# %date{HH:mm:ss} [%thread] %-5level %logger #%newline%message%newline" /> </layout> <filter type="log4net.Filter.LevelRangeFilter"> <param name="LevelMin" value="DEBUG" /> <param name="LevelMax" value="FATAL" /> </filter> </appender> <root> <!-- OFF < FATAL < ERROR < WARN < INFO < DEBUG < ALL --> <level value="ALL" /> <appender-ref ref="InfoLog" /> <appender-ref ref="DebugLog" /> <!-- <appender-ref ref="ColoredConsoleAppender" /> --> </root> </log4net> </configuration>
The above is the detailed content of Detailed examples of log4Net high-performance writing and CSV format. For more information, please follow other related articles on the PHP Chinese website!

如何使用C#编写时间序列预测算法时间序列预测是一种通过分析过去的数据来预测未来数据趋势的方法。它在很多领域,如金融、销售和天气预报中有广泛的应用。在本文中,我们将介绍如何使用C#编写时间序列预测算法,并附上具体的代码示例。数据准备在进行时间序列预测之前,首先需要准备好数据。一般来说,时间序列数据应该具有足够的长度,并且是按照时间顺序排列的。你可以从数据库或者

如何使用Redis和C#开发分布式事务功能引言分布式系统的开发中,事务处理是一项非常重要的功能。事务处理能够保证在分布式系统中的一系列操作要么全部成功,要么全部回滚。Redis是一种高性能的键值存储数据库,而C#是一种广泛应用于开发分布式系统的编程语言。本文将介绍如何使用Redis和C#来实现分布式事务功能,并提供具体代码示例。I.Redis事务Redis

如何实现C#中的人脸识别算法人脸识别算法是计算机视觉领域中的一个重要研究方向,它可以用于识别和验证人脸,广泛应用于安全监控、人脸支付、人脸解锁等领域。在本文中,我们将介绍如何使用C#来实现人脸识别算法,并提供具体的代码示例。实现人脸识别算法的第一步是获取图像数据。在C#中,我们可以使用EmguCV库(OpenCV的C#封装)来处理图像。首先,我们需要在项目

如何使用C#编写动态规划算法摘要:动态规划是求解最优化问题的一种常用算法,适用于多种场景。本文将介绍如何使用C#编写动态规划算法,并提供具体的代码示例。一、什么是动态规划算法动态规划(DynamicProgramming,简称DP)是一种用来求解具有重叠子问题和最优子结构性质的问题的算法思想。动态规划将问题分解成若干个子问题来求解,通过记录每个子问题的解,

Redis在C#开发中的应用:如何实现高效的缓存更新引言:在Web开发中,缓存是提高系统性能的常用手段之一。而Redis作为一款高性能的Key-Value存储系统,能够提供快速的缓存操作,为我们的应用带来了不少便利。本文将介绍如何在C#开发中使用Redis,实现高效的缓存更新。Redis的安装与配置在开始之前,我们需要先安装Redis并进行相应的配置。你可以

C#开发中如何处理跨域请求和安全性问题在现代的网络应用开发中,跨域请求和安全性问题是开发人员经常面临的挑战。为了提供更好的用户体验和功能,应用程序经常需要与其他域或服务器进行交互。然而,浏览器的同源策略导致了这些跨域请求被阻止,因此需要采取一些措施来处理跨域请求。同时,为了保证数据的安全性,开发人员还需要考虑一些安全性问题。本文将探讨C#开发中如何处理跨域请

如何实现C#中的图像压缩算法摘要:图像压缩是图像处理领域中的一个重要研究方向,本文将介绍在C#中实现图像压缩的算法,并给出相应的代码示例。引言:随着数字图像的广泛应用,图像压缩成为了图像处理中的重要环节。压缩能够减小存储空间和传输带宽,并能提高图像处理的效率。在C#语言中,我们可以通过使用各种图像压缩算法来实现对图像的压缩。本文将介绍两种常见的图像压缩算法:

如何在C#中实现遗传算法引言:遗传算法是一种模拟自然选择和基因遗传机制的优化算法,其主要思想是通过模拟生物进化的过程来搜索最优解。在计算机科学领域,遗传算法被广泛应用于优化问题的解决,例如机器学习、参数优化、组合优化等。本文将介绍如何在C#中实现遗传算法,并提供具体的代码示例。一、遗传算法的基本原理遗传算法通过使用编码表示解空间中的候选解,并利用选择、交叉和


Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

SublimeText3 Chinese version
Chinese version, very easy to use

SublimeText3 Mac version
God-level code editing software (SublimeText3)

MantisBT
Mantis is an easy-to-deploy web-based defect tracking tool designed to aid in product defect tracking. It requires PHP, MySQL and a web server. Check out our demo and hosting services.

Dreamweaver CS6
Visual web development tools

DVWA
Damn Vulnerable Web App (DVWA) is a PHP/MySQL web application that is very vulnerable. Its main goals are to be an aid for security professionals to test their skills and tools in a legal environment, to help web developers better understand the process of securing web applications, and to help teachers/students teach/learn in a classroom environment Web application security. The goal of DVWA is to practice some of the most common web vulnerabilities through a simple and straightforward interface, with varying degrees of difficulty. Please note that this software
