C#实现线程安全的简易日志记录方法
作者:shichen2014 发布时间:2023-12-20 22:45:17
标签:C#,线程,安全,日志,记录,方法
一般在实际项目的开发中,会要求涉及日志记录的问题,比较常用的有Log4Net,NLog等几个,而小项目小工具的话,则无需费此大驾。而譬如串口开发的话,需要记录串口过来的数据等等,这时候就要考虑日志记录上线程的问题。对此,为了方便后续使用,封装了下代码:
using System;
using System.Diagnostics;
using System.IO;
using System.Text;
using System.Threading;
namespace CSharpUtilHelpV2
{
/// <summary>
/// 日志类型枚举
/// </summary>
public enum LogType
{
/// <summary>
/// 一般输出
/// </summary>
Trace,
/// <summary>
/// 警告
/// </summary>
Warning,
/// <summary>
/// 错误
/// </summary>
Error,
/// <summary>
/// SQL
/// </summary>
SQL
}
/// <summary>
/// 基于.NET 2.0日志工具类
/// </summary>
public class LogToolV2
{
private static readonly Thread LogTask;
private static readonly ThreadSafeQueueV2<string> LogColQueue;//自定义线程安全的Queue
private static readonly object SyncRoot;
private static readonly string FilePath;
private static readonly long BackFileSize_MB = 2;//超过2M就开始备份日志文件
static LogToolV2()
{
SyncRoot = new object();
FilePath = AppDomain.CurrentDomain.SetupInformation.ApplicationBase + "Log\\";
LogTask = new Thread(WriteLog);
LogColQueue = new ThreadSafeQueueV2<string>();
LogTask.Start();
Debug.WriteLine("Log Start......");
}
/// <summary>
/// 记录日志
/// </summary>
/// <param name="msg">日志内容</param>
public static void Log(string msg)
{
string _msg = string.Format("{0} : {2}", DateTime.Now.ToString("HH:mm:ss"), msg);
LogColQueue.Enqueue(msg);
}
/// <summary>
/// 记录日志
/// </summary>
/// <param name="msg">日志内容</param>
/// <param name="type">日志类型</param>
public static void Log(string msg, LogType type)
{
string _msg = string.Format("{0} {1}: {2}", DateTime.Now.ToString("HH:mm:ss"), type, msg);
LogColQueue.Enqueue(_msg);
}
/// <summary>
/// 记录日志
/// </summary>
/// <param name="ex">异常</param>
public static void Log(Exception ex)
{
if (ex != null)
{
string _newLine = Environment.NewLine;
StringBuilder _builder = new StringBuilder();
_builder.AppendFormat("{0}: {1}{2}", DateTime.Now.ToString("HH:mm:ss"), ex.Message, _newLine);
_builder.AppendFormat("{0}{1}", ex.GetType(), _newLine);
_builder.AppendFormat("{0}{1}", ex.Source, _newLine);
_builder.AppendFormat("{0}{1}", ex.TargetSite, _newLine);
_builder.AppendFormat("{0}{1}", ex.StackTrace, _newLine);
LogColQueue.Enqueue(_builder.ToString());
}
}
private static void WriteLog()
{
while (true)
{
if (LogColQueue.Count() > 0)
{
string _msg = LogColQueue.Dequeue();
Monitor.Enter(SyncRoot);
if (!CreateDirectory()) continue;
string _path = string.Format("{0}{1}.log", FilePath, DateTime.Now.ToString("yyyyMMdd"));
Monitor.Exit(SyncRoot);
lock (SyncRoot)
{
if (CreateFile(_path))
ProcessWriteLog(_path, _msg);//写入日志到文本
}
ProcessBackLog(_path);//日志备份
}
}
}
private static void ProcessBackLog(string path)
{
lock (SyncRoot)
{
if (FileToolV2.GetMBSize(path) > BackFileSize_MB)
{
FileToolV2.CopyToBak(path);
}
}
}
private static void ProcessWriteLog(string path, string msg)
{
try
{
StreamWriter _sw = File.AppendText(path);
_sw.WriteLine(msg);
_sw.Flush();
_sw.Close();
}
catch (Exception ex)
{
Debug.WriteLine(string.Format("写入日志失败,原因:{0}", ex.Message));
}
}
private static bool CreateFile(string path)
{
bool _result = true;
try
{
if (!File.Exists(path))
{
FileStream _files = File.Create(path);
_files.Close();
}
}
catch (Exception)
{
_result = false;
}
return _result;
}
private static bool CreateDirectory()
{
bool _result = true;
try
{
if (!Directory.Exists(FilePath))
{
Directory.CreateDirectory(FilePath);
}
}
catch (Exception)
{
_result = false;
}
return _result;
}
}
}
测试代码如下:
using CSharpUtilHelpV2;
using System;
using System.Diagnostics;
using System.Threading;
namespace LogUtilHelpV2Test
{
class Program
{
static void Main(string[] args)
{
try
{
Debug.WriteLine("-------------");
Action _writeLog = delegate()
{
for (int i = 0; i < 10000; i++)
LogToolV2.Log(Guid.NewGuid().ToString(), LogType.Trace);
};
Thread _wireteLogTask1 = new Thread(new ThreadStart(_writeLog));
_wireteLogTask1.Start();
Thread _wireteLogTask2 = new Thread(new ThreadStart(_writeLog));
_wireteLogTask2.Start();
//throw new Exception("test aaa bb cc");
}
catch (Exception ex)
{
LogToolV2.Log(ex);
Console.WriteLine(ex.Message.Trim());
}
finally
{
Console.WriteLine("ok");
Console.ReadLine();
}
}
}
}
代码运行效果如下所示:
感兴趣的读者可以自己测试运行一下,希望能对大家起到一点帮助!
0
投稿
猜你喜欢
- 1、深度总结引用一位网友的话,说的非常好,如果别人问你static的作用;如果你说静态修饰 类的属性 和 类的方法 别人认为你是合格的;如果
- 一,项目简介经过调查研究进行开发设计的这款仓库管理系统,主要是为商家提供商品货物进销存的信息化管理,以便让商家在竞争如此激烈的今天占据一定的
- 本文实例为大家分享了java代码获取新浪微博应用的access token的具体代码,供大家参考,具体内容如下package test;im
- 一、Lambda 表达式的基础语 * ambda 表达式的基础语法:Java8中引入了一个新的操作符 "->" 该操
- 本文是作者结合网上的一些资料封装的java图片处理类,支持图片的缩放,旋转,马赛克化。不多说,上代码:package deal;import
- 本文介绍了spring整合JMS实现同步收发消息(基于ActiveMQ的实现),分享给大家,具体如下:1. 安装ActiveMQ注意:JDK
- SpringMVC RESTFul访问首页实现一、新建 index.html在 webapp\WEB-INF\templates 下新建首页
- The error simply says, “you've made changes in files in your works
- 一、什么是组合模式定义:将对象以树形结构组织起来,以达成“部分-整体”的层次结构,使得客户端对单个对象和组合对象的使用具有一致性。动机(Mo
- float是单精度类型,精度是8位有效数字,取值范围是10的-38次方到10的38次方,float占用4个字节的存储空间double是双精度
- C#判断数据类型的简单示例代码:int i = 5; Console
- 布局中EditText在android布局中经常用到,对EditText中输入的内容也经常需要进行限制,我们可以通过TextWatcher去
- 我们经常会看到有一些app的banner界面可以实现循环播放多个广告图片和手动滑动循环的效果。看到那样的效果,相信大家都会想到ViewPag
- WPF 之列表分页控件框架使用大于等于.NET40。Visual Studio 2022。项目使用 MIT 开源许可协议
- 前言图形相交检测常常用在伤害判定,使用自定义的图形相交检测,可以在一定程度上控制性能。比如2D格斗游戏中使用的矩形包围盒(AABB),一些动
- 前言事务对java开发的同学来说并不陌生,我们使用事务的目的在于避免产生重复数据或者说利用数据存储中间件的事务特性确保数据的精准性,比如大家
- 一:hibernate-validator 基础1. 简介:通过使用注解Annotations 给类或者类的属性加上约束(constrain
- 由于是多态对象,基类类型的变量可以保存派生类型。 要访问派生类型的实例成员,必须将值强制转换 * 生类型。 但是,强制转换会引发 Invali
- 在实践中,项目的某些配置信息是需要进行加密处理的,以减少敏感信息泄露的风险。比如,在使用Druid时,就可以基于它提供的公私钥加密方式对数据
- mybatis group by substr传参报错报异常### Cause: java.sql.SQLSyntaxErrorExcept