我正在尝试为C#类实现字符串索引器,但是当您设置属性时,字典将被设置而不是属性。这可能是我想念的简单事情,我只是看不到它。
objFiveProp temp = new objFiveProp();
temp["index1"] = 3;
将temp._items [“index1”]。值设置为3。
类别:
public class objFiveProp
{
#region Properties
private Dictionary<string, int> _items;
public int this[string key]
{
get { return _items[key]; }
set { _items[key] = value; }
}
public int index1 { get; set; }
public int index2 { get; set; }
public int index3 { get; set; }
public int index4 { get; set; }
public int index5 { get; set; }
#endregion
#region Constructor
public objFiveProp()
{
index1 = 0;
index2 = 0;
index3 = 0;
index4 = 0;
index5 = 0;
_items = new Dictionary<string, int>();
_items.Add("index1", index1);
_items.Add("index2", index2);
_items.Add("index3", index3);
_items.Add("index4", index4);
_items.Add("index5", index5);
}
#endregion
}
答案 0 :(得分:2)
这就是它的工作原理。字典包含用于设置它的整数的副本 - 而不是对属性的引用。
我会通过使用类似的东西解决这个问题:
public class objFiveProp
{
private Dictionary<string, int> _items;
public int this[string key]
{
get { return _items[key]; }
set { _items[key] = value; }
}
public int Index1
{
get { return this["index1"]; }
set { this["index1"] = value; }
}
public int Index2
{
get { return this["index2"]; }
set { this["index2"] = value; }
}
// ....
public objFiveProp()
{
_items = new Dictionary<string, int>();
_items.Add("index1", index1);
_items.Add("index2", index2);
_items.Add("index3", index3);
_items.Add("index4", index4);
_items.Add("index5", index5);
}
#endregion
这会导致您的属性始终提取存储在字典中的值,并保存在那里,因此没有两个值的副本。
答案 1 :(得分:0)
int
是值类型,而不是引用类型。在_items
中设置值时,即使您最初是从属性中添加的,也不会设置该属性。
来自MSDN
基于值类型的变量直接包含值。 将一个值类型变量分配给另一个值复制包含的值 值。这与引用类型变量的赋值不同, 它复制对象的引用,但不复制对象本身。
如果您确实需要能够从索引器和属性访问您的数据,那么最简单的方法之一就是重写您的属性:
public int indexN
{
get { return _items["indexN"]; }
set { _items["indexN"] = value; }
}
另一种方法是在索引器的setter中使用反射:
public int this[string key]
{
get { return _items[key]; }
set
{
_items[key] = value;
PropertyInfo prop = this.GetType().GetProperty(key);
if (prop != null)
{
prop.SetValue(this, null);
}
}
}
请记住,反射相对非常S-L-O-W 。
还有其他方法可以完成你想要做的事情,但也许最好的解决办法就是不要这样做。为您的班级选择最佳界面,无论是索引器还是属性,并坚持使用它。您的代码将更易于维护(您不必维护类的数据的两个公共接口)并且更具可读性(其他编码器不需要知道索引器和属性是相同的)。干杯!
答案 2 :(得分:0)
这是因为在索引集方法中,您要设置字典项
的值public int this[string key]
{
get { return _items[key]; }
set { _items[key] = value; } //here you are setting the value of dictionary item not the property
}
要么为index1,index2等创建单独的属性,要么在上面的set方法中添加检查,一个脏解决方案,根据key的值设置成员变量的值;类似的东西:
set {
_items[key] = value;
if(key == "index1")
index1 = value;
}