IDictionary赋值快捷键编译器功能或语言功能?

时间:2012-05-03 20:45:17

标签: c# .net compiler-construction

通过今天的一些随机对象创建,我遇到了Dictionary<K, V>这个简洁的小捷径。以下分配是编译器快捷方式还是Dictionary<string, string>的功能。

IDictionary<string, string> items = { { "item1key", "item1value" } };

查看Dictionary<K, V>的来源我没有看到任何关于其如何工作的方法。实现此类点的所有接口不允许我执行类似的操作。为什么我们可以为字典而不是其他类型。例如,编译器或语言功能如何知道第一项是键,第二项是值。或者甚至更具体地说,这个相同的语法不能用于List<string>

List<string> items = { "item1" };

所以第一个是有效的,为什么?

我不一定试图复制这个,而是对为什么这就是它的方式很好奇。在这种情况下,字典的特殊之处是什么?

有效的示例

public class Button
{
    public string Title { get; set; }
    public ButtonType Type { get; set; }
    public IDictionary<string, string> Items { get; set; }
    public bool RequiresSelected { get; set; }
}

var buttons = new List<Button>
    {
        new Button { 
            Items = {
                        {"button1", "Button 1"},
                        {"button2", "Button 2"},
                        {"button3", "Button 3"},
                    }, 
            Title = "3 Buttons", 
            Type = ButtonType.DropDown 
        }
    };

2 个答案:

答案 0 :(得分:7)

您显示的语法在C#中无效。你需要:

IDictionary<string, string> items = new Dictionary<string, string>
    { { "item1key", "item1value" } };

此时它只是一个普通的集合初始值设定项,因此列表等价物为:

List<string> items = new List<string> { "item1" };

编辑:让我们看看我的编辑是否可以击败你的编辑。我的猜测是你见过类似的东西:

var foo = new Foo {
    SomeDictionaryProperty = { 
         { "item1key", "item1value" }
    }
};

这是嵌入式集合初始值设定项,也可用于列表。它不会创建 new 字典,而是添加到现有字典中。上面的代码相当于:

var tmp = new Foo();
tmp.SomeDictionaryProperty.Add("item1key", "item1value");
var foo = tmp;

另一个工作的例子:

var form = new Form {
    Controls = { new Label { Text = "Foo"}, new Label { Text = "Bar" } }
};

有关详细信息,请参阅C#4规范(对象初始化程序)的7.6.10.2节。重要的是这个:

member-initializer:
    identifier   =   initializer-value

initializer-value:
    expression
    object-or-collection-initializer

因此,您可以通过对象/集合初始值设定项初始化属性 是一个特定值(在这种情况下将使用setter),在这种情况下将使用属性的 getter ,然后将setter或Add方法用于对象/集合初始值设定项的主体。

答案 1 :(得分:2)

这是C#编译器的一个特性,字典并不特殊:任何支持Add的集合都可以这种方式初始化。详情见C# Language Specification 4.0

的7.6.10.3节
  

应用集合初始值设定项的集合对象必须是实现System.Collections.IEnumerable的类型,否则会发生编译时错误。对于按顺序的每个指定元素,集合初始值设定项在目标对象上调用Add方法,并将元素初始值设定项的表达式列表作为参数列表,为每次调用应用正常的重载决策。因此,集合对象必须包含每个元素初始值设定项的适用Add方法。