繁体   English   中英

物业KeyValuePair <TKey, Tvalue> .Value没有二传手

[英]The property KeyValuePair<TKey, Tvalue>.Value has no setter

我正在使用Dictionary<int, KeyValuePair<bool, int>>来保存数据。

我不时需要在KeyValuePair增加int ,但它不会让我,因为它没有setter。 有没有办法增加它?

代码示例:

Dictionary<int, KeyValuePair<bool, int>> mDictionary = 
    new Dictionary<int, KeyValuePair<bool, int>>();

mDictionary[trapType].Value++;
//Error: The property KeyValuePair<TKey, Tvalue>>.Value has no setter

有没有办法增加它?

KeyValuePair是不可变的 - 它也是一个值类型,因此在创建副本后更改Value属性的Value无论如何都无济于事。

你必须写这样的东西:

var existingValue = mDictionary[trapType];
var newValue = new KeyValuePair<bool, int>(existingValue.Key,
                                           existingValue.Value + 1);
mDictionary[trapType] = newValue;

这很丑陋 - 你真的需要价值成为KeyValuePair吗?

mDictionary[trapType] = new KeyValuePair<bool, int>(mDictionary[trapType].Key, mDictionary[trapType].Value+1)

KeyValuePair往往只是用作字典的迭代器值。 如果你希望它是可变的,你最好定义自己的类型。

您需要构造一个新的键值对。

mDictionary[trapType] = new KeyValuePair<bool,int>(mDictionary[trapType].Key,mDictionary[trapType].Value +1);

但是,如果您使用的是.NET 4,我建议您使用Tuple而不是键值对,如果您要做的只是拥有一个包含两个其他值的对象。 即如果两者之间没有键值关系。 然后代码将是:

mDictionary[trapType] = Tuple.Create(mDictionary[trapType].Item1,mDictionary[trapType].Item2 +1);

另一个解决方案是创建一个新的字典并将新的值和键传递给它:

foreach (var d in existingDic)
                    newDic.Add(d.Key, d.Value + 1);

从概念上讲, KeyValuePair<TKey,TValue>只是一对变量, KeyValue 如果微软用暴露的公共字段KeyValue实现它,它的语义就会很完美; 当用作ForEach控制变量时,它本来是不可变的,但是可以更新普通变量或数组元素的Key或Value字段,而不必更新另一个。

遗憾的是,微软似乎一直不愿意让框架类型暴露任何公共字段,即使是像KeyValuePair<TKey,TValue>Drawing.Rectangle这样的类型,其语义规定(1)类型必须是结构; (2)其整个状态在一组固定的属性中可见(除了定义对象状态的属性之外,可能还有其他计算属性),以及(3)可以为这些属性分配适合其类型的任何值组合。 因此,Microsoft仅考虑了将包含类型状态的成员公开为只读属性或读写属性的可能性。 使用读写属性意味着代码如下:

for each (var item in myDictionary)
    item.Value += 1;

要么

...assuming MyList is a List<Drawing.Rectangle>
MyList[3].Width += 9;

将被现有的C#编译器解释为

for each (var item in myDictionary)
{ var temp = item; temp .Value += 1; }

要么

...assuming MyList is a List<Drawing.Rectangle>
{ temp = MyList[3]; temp.Width += 9; }

产生可怕的行为几乎肯定不是程序员的意图。 .net的实施者认为让KeyValuePair<TKey,TValue>的成员具有个体可变性的价值并不能证明前者所构成的危险,但修改Rectangle个体成员的有用性足以证明由此造成的危险。第二。 请注意,如果类型使用公开的字段而不是属性,则两个示例都不必构成任何危险,因为即使在调用属性setter时,也不允许写入临时结构的字段。

如前所述,KeyValuePair是不可变的。 我认为值得在这里添加一个可变的实现:

public class MutableKeyValuePair<KeyType, ValueType>
    {
        public KeyType Key { get; set; }
        public ValueType Value { get; set; }

        public MutableKeyValuePair() { }

        public MutableKeyValuePair(KeyType key, ValueType val)
        {
            Key = key;
            Value = val;
        }
    }

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM