C# 实现写日志(支持多线程读写)

     在写入一些内容到某个日志文件中,在另一个进程/线程中要读取文件内容的时候报异常,提示 System.IO.IOException: 文件“xxxxxx”正由另一进程使用,因此该进程无法访问此文件。

    FileShare 枚举:

    ​包含用于控制其他 FileStream 对象对同一文件可以具有的访问类型的常数。

 使用FileShare.ReadWrite属性可解决多线程时,写入日志报异常问题。

Log.cs

using Newtonsoft.Json;
using System;
using System.IO;
using System.Text;

namespace Test
{
    public class Log
    {
        private static string log_dir = AppDomain.CurrentDomain.BaseDirectory + "LOG";
        private static string warn_dir = AppDomain.CurrentDomain.BaseDirectory + "WARN_LOG";
        private static string log_path
        {
            get
            {
                if (!Directory.Exists(log_dir))
                {
                    Directory.CreateDirectory(log_dir);
                }
                if (!Directory.Exists(warn_dir))
                {
                    Directory.CreateDirectory(warn_dir);
                }
                return DateTime.Now.ToString("yyyyMMddHH");
            }
        }
        private static string warn_path { get { return warn_dir + "\\" + log_path + ".txt"; } }
        private static string path { get { return log_dir + "\\" + log_path + ".txt"; } }
        public static void write_log(string str)
        {
            string strlog = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + ": " + str + "\r\n";

            using (FileStream fs = new FileStream(path, FileMode.Append, FileAccess.Write, FileShare.ReadWrite))
            {
                using (StreamWriter sw = new StreamWriter(fs))
                {
                    sw.Write(strlog);
                    sw.Flush();
                    sw.Close();
                    fs.Close();
                }
            }
     //       File.AppendAllText(path, DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + ": " + str + "\r\n");

        }
        public static void write_warn_log(string str)
        {
            string strlog = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + ": " + str + "\r\n";

            using (FileStream fs = new FileStream(path, FileMode.Append, FileAccess.Write, FileShare.ReadWrite))
            {
                using (StreamWriter sw = new StreamWriter(fs))
                {
                    sw.Write(strlog);
                    sw.Flush();
                    sw.Close();
                    fs.Close();
                }
            }
      //      File.AppendAllText(warn_path, DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + ": " + str + "\r\n");
        }

        public static void write_json(string title,string json)
        {
            try
            {

                JsonSerializer serializer = new JsonSerializer();
                JsonTextReader jtr = new JsonTextReader(new StringReader(json));
                object obj = serializer.Deserialize(jtr);
                if (obj != null)
                {
                    StringWriter textWriter = new StringWriter();
                    JsonTextWriter jsonWriter = new JsonTextWriter(textWriter)
                    {
                        Formatting = Formatting.Indented,
                        Indentation = 4,//缩进字符数
                        IndentChar = ' '//缩进字符
                    };
                    serializer.Serialize(jsonWriter, obj);
                    Log.write_log(title  + textWriter.ToString());
                }
            }
            catch (Exception ex)
            {
                Log.write_log(title + ", 错误的JSON:" + ex.Message);
            }
        }
    }
}

调用:

Log.write_log("");
Log.write_warn_log("");
Log.write_json("");

  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C# 的 WebForm 项目中,可以使用多线程实现一些后台任务的并发执行,从而提高系统的性能和响应速度。下面是一个简单的示例,说明如何在 WebForm 项目中使用多线程: 1. 在代码中创建一个新的线程,用于执行后台任务: ```csharp Thread t = new Thread(new ThreadStart(DoWork)); t.Start(); ``` 这里的 `DoWork` 是一个自定义的方法,用于执行具体的后台任务。 2. 在 `DoWork` 方法中编具体的后台任务代码。为了避免线程冲突,需要使用锁或其他线程同步机制来保证数据的一致性。 ```csharp private void DoWork() { // 后台任务代码 lock (this) { // 保证数据的一致性 // ... } } ``` 3. 在 WebForm 页面中,可以使用异步回调的方式来处理后台任务的结果。例如,可以使用 `BeginInvoke` 方法来异步调用一个回调函数: ```csharp private void Page_Load(object sender, EventArgs e) { // 异步调用后台任务,并指定回调函数 Func<int> func = new Func<int>(DoWork); IAsyncResult result = func.BeginInvoke(new AsyncCallback(OnComplete), null); } private void OnComplete(IAsyncResult result) { // 处理后台任务的结果 int result = func.EndInvoke(result); // ... } ``` 这里的 `DoWork` 方法和之前的一样,用于执行具体的后台任务。在 `Page_Load` 方法中,使用 `BeginInvoke` 方法异步调用 `DoWork` 方法,并指定一个回调函数 `OnComplete`。当 `DoWork` 方法执行完成后,会自动调用 `OnComplete` 方法,并将执行结果传递给它。 以上就是在 C# 的 WebForm 项目中使用多线程的简单示例。需要注意的是,使用多线程时需要小心处理线程之间的数据同步和资源竞争问题,否则容易引起不可预知的错误。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值