更新 AddOrUpdate 方法中的密钥对象



我有一个测试类。

class Test
{
    public int Id { get; set; }
    public int Val { get; set; }
}

我想设置一个 ConcurrentDictionary,其中 int 作为键,测试作为值。

bool External = true;
ConcurrentDictionary<int, Test> data = new ConcurrentDictionary<int, Test>();

我想为这个字典编写 AddorUpdate 的更新部分,以便如果外部变量(例如外部变量)为 true,那么该 Test 实例的 Val 应该增加 100,但如果 bool 为 false,那么它应该减少 100。有人可以帮我怎么做。我只是不确定如何使用 lambda 访问字典中的测试实例。尽管有 lambda,我也可以进行方法调用吗?

像这样:

data.AddOrUpdate(key, test, (k, t) =>
{
    var newTest = new Test { Id = t.Id, Val = t.Val };
    if (External)
        newTest.Val += 100;
    else
        newTest.Val -= 100;
    return newTest;
});

您的示例中的bool External 最终将成为匿名方法中的闭包,因此它会变得非常奇怪并产生意想不到的结果。你会想以某种方式解决这个问题。

编辑:

我对这种方法不满意。我建议去一个常规Dictionary<int, Test>,提取当前值并更新它,所有这些都有一个ReaderWriterLockSlim来确保状态。

var key = ...;
var lock = new ReaderWriterLockSlim();
lock.EnterWriteLock();
try
{
    if (dict.ContainsKey(key))
    {
        // update without closures
        var test = dict[key];
        if (External)
            test.Val += 100;
        else
            test.Val -= 100;
    }
}
else
{
    // insert
    var test = new Test { ...initial state... };
    dict.Add(key, test);
}
finally
{
    lock.ExitWriteLock();
}

最后,请务必将External标记为volatile以创建内存屏障并防止可能给您带来过时值的优化。

最新更新