c#类索引器设置字典不是属性

本文关键字:属性 字典 设置 索引 | 更新日期: 2023-09-27 18:06:41

我正在尝试为c#类实现字符串索引器,但是当您设置属性时,字典会被设置而不是属性。可能是我缺少了什么简单的东西,只是我看不见而已。

objFiveProp temp = new objFiveProp();
temp["index1"] = 3;

设置temp._items["index1"]。

类:

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
}

c#类索引器设置字典不是属性

就是这样。Dictionary包含用于设置它的整数的副本,而不是对属性的引用。

我可以这样处理:

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

这会使您的属性始终提取存储在字典中的值,并保存在那里,因此不会有两个值的副本。

这是因为在索引集方法中,您正在设置字典项的值

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;
}

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

还有其他方法可以完成你想做的事情,但也许最好的解决方案是根本不这样做。为您的类选择最佳接口,无论是索引器还是属性,并坚持使用它。您的代码将更易于维护(您不必维护类数据的两个公共接口)并且更具可读性(其他编码人员将不需要知道索引器和属性是同一件事)。干杯!