2

我有几个在不同线程上运行的客户端对象(TCPClient 包装器)。如果这些对象中的任何一个遇到问题,则会将错误消息保存到 XML 错误日志中。显然,文件访问一次仅限于一个进程,所以我需要一种方法来防止其他线程在另一个线程使用它时读取/写入。

我目前正在使用该lock方法,但是仍然会引发另一个进程正在使用该文件的异常。我的印象是lock会管理等待和重试。

// Lock the XML IO for safety due to multi-threading
lock (this.xmlDoc) // Changed from this to the xmlDoc
{
    // Attempt to load existing xml
    try
    {
        this.xmlDoc.Load(this.logPath);
    }
    catch (FileNotFoundException e)
    {
        // xml file doesn't exist, create
        this.xmlDoc.AppendChild(this.xmlDoc.CreateElement("root"));
    }
    // Get the doc root
    XmlElement root = this.xmlDoc.DocumentElement;
    // Create message entry
    XmlElement msg = this.xmlDoc.CreateElement("message");
    // Add <time></time> to msg
    msg.AppendChild(this.xmlDoc.CreateElement("time")).InnerText = dt.ToString();
    // Add <error></error> to msg
    msg.AppendChild(this.xmlDoc.CreateElement("error")).InnerText = message;
    // Add msg to root
    root.AppendChild(msg);
    // Save. Done.
    this.xmlDoc.Save(this.logPath);
}
4

2 回答 2

3

lock不知道文件是什么。它对每个进程都存在的 CLR 对象进行操作。不同的进程将看到不同的对象。

你需要一些跨进程互斥。以下是一些选项:

  1. 一个名为Mutex
  2. 文件级锁定和重试策略(我怀疑 log4net 是这样做的)
  3. 具有不同名称的多个文件(使用随机名称,以免发生冲突)

选项 3 是最简单的。

于 2013-08-16T14:22:06.440 回答
0

如果所有线程都来自同一个进程:

使用 System.Diagnostics.Trace。它内置在框架中,包含线程安全,您可以使用配置文件来定义跟踪侦听器。

构造一个 XML 字符串,然后构造 Trace.WriteLine()。

如果您的线程来自不同的进程,则每个进程都需要维护自己的日志文件,正如@usr 提到的那样。

作为旁注,如果音量足够低以避免文件锁定问题,您可以将错误记录到 Windows 事件查看器。

于 2013-08-16T14:36:57.040 回答