解决C#跨线程访问XML对象的异常 安全的并发XML处理模式

来源:这里教程网 时间:2026-02-21 17:31:59 作者:

在C#多线程编程中,直接从多个线程读写同一个XML对象(如

XmlDocument
XDocument
)容易引发异常或数据不一致问题。这是因为这些类本身不是线程安全的。要解决跨线程访问XML对象的异常,必须采用合理的并发控制策略。

使用锁机制保护共享XML对象

最直接的方式是通过

lock
语句确保同一时间只有一个线程能操作XML对象。

XDocument
为例:

private static readonly object _xmlLock = new object();
private static XDocument _document = XDocument.Load("data.xml");
public static void UpdateXml(string elementName, string value)
{
    lock (_xmlLock)
    {
        var element = _document.Root?.Element(elementName);
        if (element != null)
            element.Value = value;
        else
            _document.Root?.Add(new XElement(elementName, value));
        
        _document.Save("data.xml");
    }
}
public static string ReadXml(string elementName)
{
    lock (_xmlLock)
    {
        return _document.Root?.Element(elementName)?.Value;
    }
}

这种方式简单有效,适用于读写频率不高或并发量较小的场景。

采用不可变模式避免共享状态

为减少锁竞争,可使用“每次修改生成新文档”的方式,配合

Interlocked
volatile
字段实现线程安全更新。

示例:

private static volatile XDocument _currentDoc = XDocument.Load("data.xml");
public static void UpdateXmlSafe(string elementName, string value)
{
    XDocument oldDoc, newDoc;
    do
    {
        oldDoc = _currentDoc;
        newDoc = new XDocument(oldDoc); // 克隆当前文档
        var element = newDoc.Root?.Element(elementName);
        if (element != null)
            element.Value = value;
        else
            newDoc.Root?.Add(new XElement(elementName, value));
    } while (Interlocked.CompareExchange(ref _currentDoc, newDoc, oldDoc) != oldDoc);
    // 可选:异步保存到文件
    Task.Run(() => newDoc.Save("data.xml"));
}
public static string ReadXmlSafe(string elementName)
{
    var doc = _currentDoc; // 读取volatile引用
    return doc.Root?.Element(elementName)?.Value;
}

这种方法适合读多写少的场景,能显著降低锁争用。

使用ReaderWriterLockSlim支持高并发读取

当XML频繁被读取而偶尔更新时,

ReaderWriterLockSlim
比普通
lock
更高效。

private static ReaderWriterLockSlim _rwLock = new ReaderWriterLockSlim();
private static XDocument _sharedDoc = XDocument.Load("data.xml");
public static string ReadWithRwLock(string elementName)
{
    _rwLock.EnterReadLock();
    try
    {
        return _sharedDoc.Root?.Element(elementName)?.Value;
    }
    finally
    {
        _rwLock.ExitReadLock();
    }
}
public static void WriteWithRwLock(string elementName, string value)
{
    _rwLock.EnterWriteLock();
    try
    {
        var element = _sharedDoc.Root?.Element(elementName);
        if (element != null)
            element.Value = value;
        else
            _sharedDoc.Root?.Add(new XElement(elementName, value));
        
        _sharedDoc.Save("data.xml");
    }
    finally
    {
        _rwLock.ExitWriteLock();
    }
}

ReaderWriterLockSlim
允许多个读线程同时访问,仅在写入时阻塞所有其他操作。

基本上就这些。选择哪种模式取决于你的使用场景:低并发用

lock
,高读低写考虑
ReaderWriterLockSlim
或不可变模式。关键是避免多个线程直接操作同一个XML对象而不加同步。安全的并发处理不是难事,但需要主动设计。

相关推荐