以下代码可以编译,但失败并显示
NullReferenceException
:
class Test
{
public Dictionary<string, string> Dictionary { get; set; }
}
static void Main(string[] args)
{
var x = new Test
{
Dictionary = // fails
{
{ "key", "value" }, { "key2", "value2" }
}
};
}
如果将标记为“失败”的行替换为以下内容,则它可以工作(如预期):
Dictionary = new Dictionary<string, string>
失败的语法有什么目的吗——它可以在其他情况下成功使用吗?或者这是编译器的疏忽?
不,这不是一个错误...这是您对初始化语法的理解存在缺陷:)
的想法
Dictionary = { ... }
适用于调用者对集合属性具有 read 访问权限,但没有 write 访问权限的情况。换句话说,像这样的情况:
class Test
{
private readonly Dictionary<string, string> dictionary
= new Dictionary<string, string>();
public Dictionary<string, string> Dictionary { get { return dictionary; } }
}
基本上,它最终会调用 Add,但不会先创建新集合。所以这段代码:
Test test = new Test { Dictionary = { { "a", "b"}, {"c", "d" } };
相当于:
Test tmp = new Test();
Dictionary<string, string> tmpDictionary = tmp.Dictionary;
tmpDictionary.Add("a", "b");
tmpDictionary.Add("c", "d");
Test test = tmp;
UI 的
Controls
集合就是一个很好的例子。你可以这样做:
Form form = new Form
{
Controls =
{
new Button { Text = "Hi" },
new TextBox { Text = "There" }
}
};
但您实际上无法设置
Controls
属性,因为它是只读的。
您仍然可以在构造函数中使用您想要的语法:
Dictionary<string, string> dictionary = new Dictionary<string, string>
{
{"a", "b"},
{"c", "d"}
};
它失败并出现空引用异常,因为您声明了一个未初始化的变量(字典),因此它是空的。
当您尝试使用初始化语法向其中添加条目时,您正在尝试将数据写入空对象。
当您用“= new Dictionary...”替换该行时,您正在创建一个新对象供 Dictionary 引用,因此您可以成功向其中添加条目。
(在 Jon Skeet 的示例中,Controls 集合必须已经由 Form 创建,因此它可以正常工作)