由于程序是3层架构的,所有多线程记录日志成了比较棘手的问题,以前还真就没有在意过写日志的问题,认为不过是写文件罢了~~!如今发现原来要实现文件共享,并且能够使多线程同时操作日志还不能相互冲突,真的很麻烦。当然要实现它我首先想到的是在网上搜,结果可能是我搜的不得其法,没发现结果,多数都是用lock,mutx等线程锁或互斥的方式写日志,偶想这样和单线程有啥区别吗?还是没能起到多线程应该有的效率!

后来问朋友,发现个log4net的东西,不过此物依然用到了线程互斥,看了源码发现的!

网络不行,朋友不知道,只好自己想办法。想了几种方法如下:

1、写多个文件,然后找个机会把这些文件合并!

2、干脆放弃写文件,改写数据库!

3、把文件看成一个表结构,实现行级锁。也就是一个线程写一行。

4、把文件看成内存块,每块写完再合并!

以上几种方法缺点:

1、多个文件。。。想起来就恶心!

2、写数据库?太没技术含量咱不干!

3、行级锁?我一个线程要写入多行不是看起来很乱?

4、没想到啥缺点,实现它吧!

以下是偶经过了10000线程并发测试,也没发现问题的代码!

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. namespace MTW
  5. {
  6. public class MTWFile
  7. {
  8. private string _fileName;
  9. private static Dictionary<long, long> lockDic = new Dictionary<long, long>();
  10. /// <summary>
  11. /// 获取或设置文件名称
  12. /// </summary>
  13. public string FileName
  14. {
  15. get { return _fileName; }
  16. set { _fileName = value; }
  17. }
  18. /// <summary>
  19. /// 构造函数
  20. /// </summary>
  21. /// <param name="byteCount">每次开辟位数大小,这个直接影响到记录文件的效率</param>
  22. /// <param name="fileName">文件全路径名</param>
  23. public MTWFile(string fileName)
  24. {
  25. _fileName = fileName;
  26. }
  27. /// <summary>
  28. /// 创建文件
  29. /// </summary>
  30. /// <param name="fileName"></param>
  31. public void Create(string fileName)
  32. {
  33. if (!System.IO.File.Exists(fileName))
  34. {
  35. using (System.IO.FileStream fs = System.IO.File.Create(fileName))
  36. {
  37. fs.Close();
  38. }
  39. }
  40. }
  41. /// <summary>
  42. /// 写入文本
  43. /// </summary>
  44. /// <param name="content">文本内容</param>
  45. private void Write(string content, string newLine)
  46. {
  47. if (string.IsNullOrEmpty(_fileName))
  48. {
  49. throw new Exception("FileName不能为空!");
  50. }
  51. using (System.IO.FileStream fs = new System.IO.FileStream(_fileName, System.IO.FileMode.OpenOrCreate, System.IO.FileAccess.ReadWrite, System.IO.FileShare.ReadWrite, 8, System.IO.FileOptions.Asynchronous))
  52. {
  53. //Byte[] dataArray = System.Text.Encoding.ASCII.GetBytes(System.DateTime.Now.ToString() + content + "/r/n");
  54. Byte[] dataArray = System.Text.Encoding.Default.GetBytes(content + newLine);
  55. bool flag = true;
  56. long slen = dataArray.Length;
  57. long len = 0;
  58. while (flag)
  59. {
  60. try
  61. {
  62. if (len >= fs.Length)
  63. {
  64. fs.Lock(len, slen);
  65. lockDic[len] = slen;
  66. flag = false;
  67. }
  68. else
  69. {
  70. len = fs.Length;
  71. }
  72. }
  73. catch (Exception ex)
  74. {
  75. while (!lockDic.ContainsKey(len))
  76. {
  77. len += lockDic[len];
  78. }
  79. }
  80. }
  81. fs.Seek(len, System.IO.SeekOrigin.Begin);
  82. fs.Write(dataArray, 0, dataArray.Length);
  83. fs.Close();
  84. }
  85. }
  86. /// <summary>
  87. /// 写入文件内容
  88. /// </summary>
  89. /// <param name="content"></param>
  90. public void WriteLine(string content)
  91. {
  92. this.Write(content, System.Environment.NewLine);
  93. }
  94. /// <summary>
  95. /// 写入文件
  96. /// </summary>
  97. /// <param name="content"></param>
  98. public void Write(string content)
  99. {
  100. this.Write(content, "");
  101. }
  102. }
  103. }

调用起来很简单,实例化,然后随便调用一个write或writeLine方法!

C#多线程写日志的更多相关文章

  1. 一种多线程写日志文件的解决方案 c#源代码演示

    using System;using System.Collections.Generic;using System.Collections;using System.Text;using Syste ...

  2. c# 多线程使用队列顺序写日志的类 (需要再优化)

    using System; using System.Collections.Generic; using System.Threading; public class LogManager { // ...

  3. Delphi 写日志的类

    unit uProgLog; interface uses Windows, SysUtils, SyncObjs; const C_LOG_LEVEL_TRACE = $; C_LOG_LEVEL_ ...

  4. 重复造轮子,编写一个轻量级的异步写日志的实用工具类(LogAsyncWriter)

    一说到写日志,大家可能推荐一堆的开源日志框架,如:Log4Net.NLog,这些日志框架确实也不错,比较强大也比较灵活,但也正因为又强大又灵活,导致我们使用他们时需要引用一些DLL,同时还要学习各种用 ...

  5. C#写日志工具类

    代码: using System; using System.Collections.Generic; using System.IO; using System.Linq; using System ...

  6. .NET Core的日志[5]:利用TraceSource写日志

    从微软推出第一个版本的.NET Framework的时候,就在“System.Diagnostics”命名空间中提供了Debug和Trace两个类帮助我们完成针对调试和跟踪信息的日志记录.在.NET ...

  7. [转]ASP.NET Core 开发-Logging 使用NLog 写日志文件

    本文转自:http://www.cnblogs.com/Leo_wl/p/5561812.html ASP.NET Core 开发-Logging 使用NLog 写日志文件. NLog 可以适用于 . ...

  8. Spring 使用 SLF4J代替 Commons Logging 写日志 异常

    项目的日志更换成slf4j和logback后,发现项目无法启动.错误提示 Caused by: java.lang.NoClassDefFoundError: Lorg/apache/commons/ ...

  9. 程序员的修养 -- 如何写日志(logging)

      在程序中写日志是一件非常重要,但是很容易被开发人员忽视的地方.写好程序的日志可以帮助我们大大减轻后期维护压力. 在实际的工作中,开发人员往往迫于的巨大时间压力,而写日志又是一个非常繁琐的事情,往往 ...

随机推荐

  1. 20172301 《Java软件结构与数据结构》实验二报告

    20172301 <Java软件结构与数据结构>实验二报告 课程:<Java软件结构与数据结构> 班级: 1723 姓名: 郭恺 学号:20172301 实验教师:王志强老师 ...

  2. JAVA 图形界面开发基础详解

    与C的win32一样,JAVA也有自己的图形界面开发,将在此篇博客中对基础部分进行讲解. 1.Java提供的图形界面类有哪些? Java提供了两套图形界面 (1)AWT组建(基础) AWT组件是jdk ...

  3. Django中的ORM关系映射查询方面

    ORM:Object-Relation Mapping:对象-关系映射 在MVC框架中的Model模块中都包括ORM,对于开发人员主要带来了如下好处: 实现了数据模型与数据库的解耦,通过简单的配置就可 ...

  4. 初识thinkphp(3)

    这篇内容主要涉及请求相应内容. 该系列主要是个人笔记,且内容是连贯的,其中涉及到的自己写的模块或者方法在前面文章中有介绍咋来的,如果您看得云里雾里,给您带来不便,真的不好意思. 0x01:请求对象 官 ...

  5. SlickOne 敏捷开发框架介绍(二) -- 多用户/多租户/SAAS软件基础框架实现

    前言:在应用于集团版客户或SAAS平台服务的业务系统中,流程管理系统需要支持多用户组织模型.其中包括角色数据.流程定义数据和流程实例数据的多用户标识绑定.本文旨在全面描述如何基于SlickOne敏捷开 ...

  6. SSH公钥登录原理

    在平时工作中我们经常要远程登录服务器,这就要用到SSH协议: $ ssh user@host 主要有两种登录方式:第一种为密码口令登录,第二种为公钥登录 密码口令登录 通过密码进行登录,主要流程为: ...

  7. spring cloud 学习(5) - config server

    分布式环境下的统一配置框架,已经有不少了,比如百度的disconf,阿里的diamand.今天来看下spring cloud对应的解决方案: 如上图,从架构上就可以看出与disconf之类的有很大不同 ...

  8. sqlserver2014内存数据库特性介绍

    sql server 2014提供了众多激动人心的新功能,但其中我想最让人期待的特性之一就要算内存数据库了,下面就简单介绍一下sql server 2014的内存数据库的一些特性   相信大家对内存数 ...

  9. 用Visio画泳道图

    在一次会议中看到有个同事在讲解业务流程时画了一个与PD中很类似的泳道图,但是在图的左侧确有一个阶段的列,事后与他沟通,才知道他这个图是”拼”出来的,也就是说所有的图都是他一点点的在画图工具中做出来的. ...

  10. The STM32 SPI and FPGA communication

    The STM32 SPI and FPGA communication STM32 spi bus communication SPI bus in the study, the protocol ...