c# - 这个非锁定的 TryGetValue() 字典访问线程安全吗?

标签 c# .net multithreading dictionary thread-safety

private object lockObj = new object();

private Dictionary<int, string> dict = new Dictionary<int, string>();

public string GetOrAddFromDict(int key)
{
    string value;

    // non-locked access:
    if (dict.TryGetValue(key, out value))
        return value;

    lock (this.lockObj)
    {
        if (dict.TryGetValue(key, out value))
            return value;

        string newValue = "value of " + key; // place long operation here
        dict.Add(key, newValue);

        return newValue;
    }
}

问题a:它是线程安全的吗?如果是,为什么?

问题 b:这个双 TryGetValue() 模式是如何调用的?

最佳答案

a) 这不是线程安全的,因为底层 Dictionary 本身不是线程安全的。如果另一个线程同时调用 Add,可能会发生未定义的行为。

b) 这实际上是对 double-checked locking 的尝试.

我建议使用 ConcurrentDictionary类,因为它是为这种情况设计的。另一种选择是使用 ReaderWriterLockSlim (或 ReaderWriterLock ),如果您的目标不是 .NET 4.0。

关于c# - 这个非锁定的 TryGetValue() 字典访问线程安全吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6738634/

相关文章:

c# - 如何获取泛型方法的类型参数?

c# - 将一个 Windows 窗体设置为另一个 Windows 窗体的 MDI 子窗体(或等同物)

c# - 从 C# 动态类型调用 `Contains` 方法会产生错误 - 为什么?

c# - 调用 new SqlConnection() 挂起程序

Java-同步

java - 处理由多个线程使用的实例中的状态

c# - 无法获取模型的反射类型

c# - 输入管理器 2 个播放器和 2 个 Controller

c# - 如何将 Nuget 包 dll 添加到 Wix 安装程序

c - 在 Solaris 上运行的此代码中信号量 func semctl() 出现总线错误