最近在使用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慢一点,因为它每次都会打开关闭文件流。关于log4net里面的变量我也不是很清楚,不过有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 > && _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="&quot;%date{yyyy-MM-dd HH:mm:ss}&quot;,&quot;%message%&quot; "/>
</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="" />
<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="" />
<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="&quot;%date{yyyy-MM-dd HH:mm:ss}&quot;,&quot;%message%&quot; "/>
</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>

log4Net 高性能写入和CSV格式的更多相关文章

  1. python 将json格式的数据写入csv格式的文件中

    # coding=utf-8 import json import csv # 重新进行配置读写数据时的默认编码 import sys reload(sys) sys.setdefaultencodi ...

  2. Python数据写入csv格式文件

    (只是传递,基础知识也是根基) Python读取数据,并存入Excel打开的CSV格式文件内! 这里需要用到bs4,csv,codecs,os模块. 废话不多说,直接写代码!该重要的内容都已经注释了, ...

  3. python3 库pandas写入csv格式文件出现中文乱码问题解决方法

    python3 库pandas写入csv格式文件出现中文乱码问题解决方法 解决方案: 问题是使用pandas的DataFrame的to_csv方法实现csv文件输出,但是遇到中文乱码问题,已验证的正确 ...

  4. 假期学习【十一】Python爬取百度词条写入csv格式 python 2020.2.10

    今天主要完成了根据爬取的txt文档,从百度分类从信息科学类爬取百度词条信息,并写入CSV格式文件. txt格式文件如图: 为自己爬取内容分词后的结果. 代码如下: import requests fr ...

  5. C#对.CSV格式的文件--逗号分隔值文件 的读写操作及上传ftp服务器操作方法总结

    前言 公司最近开发需要将数据保存到.csv文件(逗号分隔值 文件)中然后上传到ftp服务器上,供我们系统还有客户系统调用,之前完全没有接触过这个,所以先来看看百度的解释:逗号分隔值(Comma-Sep ...

  6. php生成CSV格式(转)

    参考网址: php对csv文件的常用操作集合 http://blog.xhbin.com/archives/748 1,下载CSV格式文档 唯一需要特别注意的是编码. <? include_on ...

  7. 批量处理csv格式转换成xls

    结合下面的代码学习相关模块及函数方法的使用 #coding:utf-8 #导入相应模块 import csv import xlwt import sys import os import fnmat ...

  8. csv格式订单下载,完成后伴随邮件通知下载

    前言 功能开发中会遇到大量订单下载,而服务器的请求响应时间又配置的很短,导致下载时候请求超时. 这篇文章主要思路:异步查询数据,生成csv文件,放入email中并发送给用户.(异步部分本文不做介绍,配 ...

  9. R语言︱用excel VBA把xlsx批量转化为csv格式

    笔者寄语:批量读取目前看到有以下几种方法:xlsx包.RODBC包.批量转化成csv后读入.本章来自博客:http://www.cnblogs.com/weibaar/p/4506144.html 在 ...

随机推荐

  1. C++ code:prime decision

    1 判断一个数是否为素数 对于判断一个数m是否为素数,最朴素的方式是按照素数的定义,试除以从2开始到m-1的整数,倘若无一例外地不能整除,则该数必为素数. #include<iostream&g ...

  2. jquery.Inputmask 插件用法(中文API文档)

      jquery.Inputmask 可以算是input文本输入限制的神器了,内部融合了多种输入限制, 如金额,电话号码,身份证号,网关等..,并且还可以自定义规则. inputmask 据说最早起源 ...

  3. 《Java编程的逻辑》 - 文章列表

    <计算机程序的思维逻辑>系列文章已整理成书<Java编程的逻辑>,由机械工业出版社出版,2018年1月上市,各大网店有售,敬请关注! 京东自营链接:https://item.j ...

  4. linux 中的./configuration --prefix=安装路径 的用法(指定源码安装方式的安装路基)

    源码的安装一般由3个步骤组成:配置(configure).编译(make).安装(make install). Configure是一个可执行脚本,它有很多选项,在待安装的源码路径下使用命令./con ...

  5. BZOJ1297 [SCOI2009]迷路 矩阵乘法

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1297 题意概括 有向图有 N 个节点,从节点 0 出发,他必须恰好在 T 时刻到达节点 N-1. ...

  6. Topic路由模式

    原本理解的不够彻底,程序总是不太对.所以查询了资料,关于这种模式的意思做了仔细的解读,复制的文字. 一:介绍 1.模式 2.知识点 其中,#可以匹配一个或者多个字符 其中,*可以匹配一个字符 3.仔细 ...

  7. PHP 随笔记

    SQL插入数据并返回刚插入数据行的主键ID INSERT INTO `peoplespublic`.`demo` (`id`, `name`, `content`, `time`, `file`) V ...

  8. cache 基本原理

        Cache 主要由 Cache Tag,Cache 存储体,Cache 控制模块组成.Cache Tag 主要用来记录 Cache 存储体中数据的位置和判断 Cache 内数据是否命中: Ca ...

  9. hdu1873 看病要排队【优先队列】

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1873 看病要排队 Time Limit: 3000/1000 MS (Java/Others)     ...

  10. 007.基于Docker的Etcd分布式部署

    一 环境准备 1.1 基础环境 ntp配置:略 #建议配置ntp服务,保证时间一致性 etcd版本:v3.3.9 防火墙及SELinux:关闭防火墙和SELinux 名称 地址 主机名 备注 etcd ...