两者之间有什么区别?何时使用“对象初始值设定项”而不是“构造函数”,反之亦然?我正在使用C#,如果这很重要的话。此外,对象初始化方法是否特定于C#或.NET?
答案 0 :(得分:220)
对象初始化器是添加到C#3的东西,以便在使用对象时简化对象的构造。
构造函数在给定0个或更多参数的情况下运行,并用于在调用方法获取创建对象的句柄之前创建和初始化对象。例如:
MyObject myObjectInstance = new MyObject(param1, param2);
在这种情况下,MyObject
的构造函数将使用值param1
和param2
运行。这些都用于在内存中创建新的MyObject
。返回创建的对象(使用这些参数设置),并设置为myObjectInstance
。
通常,为了完全设置一个对象,让构造函数需要所需的参数被认为是一种好习惯,因此无法在无效状态下创建对象。
但是,通常可以设置“额外”属性,但不是必需的。这可以通过重载的构造函数来处理,但是会导致有很多构造函数在大多数情况下都不一定有用。
这会导致对象初始值设定项 - 对象初始化程序允许您在构建之后设置对象上的属性或字段,但之前可以通过其他任何方式使用它。例如:
MyObject myObjectInstance = new MyObject(param1, param2)
{
MyProperty = someUsefulValue
};
这与您执行此操作的行为大致相同:
MyObject myObjectInstance = new MyObject(param1, param2);
myObjectInstance.MyProperty = someUsefulValue;
但是,在多线程环境中,对象初始值设定项的原子性可能是有益的,因为它可以防止对象处于未完全初始化状态(有关更多信息,请参阅this answer细节) - 它可以像你想象的那样为null或初始化。
此外,对象初始值设定项更易于阅读(特别是在设置多个值时),因此它们可以为构造函数提供与许多重载相同的好处,而无需使许多重载使该类的API复杂化。 p>
答案 1 :(得分:43)
构造函数是类型上的已定义方法,它采用指定数量的参数并用于创建和初始化对象。
对象初始值设定项是在构造函数之后在对象上运行的代码,可用于将对象上的任意数量的字段简洁地设置为指定值。在调用构造函数之后,会设置这些字段。
如果构造函数充分设置了对象的初始状态,则可以在没有对象初始值设定项的帮助下使用构造函数。但是,对象初始值设定项必须与构造函数一起使用。语法要求构造函数的显式或隐式使用(VB.Net和C#)来创建初始对象。当构造函数没有充分初始化对象并使用一些简单的字段和/或属性集时,您将使用对象初始值设定项。
答案 2 :(得分:24)
当你这样做时
Person p = new Person { Name = "a", Age = 23 };
这是对象初始化器基本上做的事情:
Person tmp = new Person(); //creates temp object calling default constructor
tmp.Name = "a";
tmp.Age = 23;
p = tmp;
现在这有助于this之类的行为。了解对象初始化程序的工作方式非常重要。
答案 3 :(得分:14)
如果你的对象必须设置属性以使其正常工作,一种方法是只公开一个构造函数,它需要那些必需的属性作为参数。
在这种情况下,如果不指定这些必需属性,则无法创建对象。对象初始化器无法强制执行类似的操作。
对象初始化器实际上只是缩短初始分配的“语法便利性”。很好,但功能上并不是很相关。
马克
答案 4 :(得分:4)
构造函数是一种方法(可能)接受参数并返回类的新实例。它可能包含初始化逻辑。 您可以在下面看到构造函数的示例。
public class Foo
{
private SomeClass s;
public Foo(string s)
{
s = new SomeClass(s);
}
}
现在考虑以下示例:
public class Foo
{
public SomeClass s { get; set; }
public Foo() {}
}
使用对象初始值设定项可以获得与第一个示例中相同的结果,假设您可以使用以下代码访问SomeClass:
new Foo() { s = new SomeClass(someString) }
正如您所看到的,对象初始值设定项允许您在执行构造的同时指定公共字段和公共(可设置)属性的值,并且在构造函数不提供初始化某些字段的任何重载时尤其有用。 请注意,然而,对象初始化器只是语法糖,并且在编译之后与分配序列的确不同。
答案 5 :(得分:0)
对象初始化器在LINQ查询表达式中特别有用。查询表达式频繁使用匿名类型,这些匿名类型只能通过使用对象初始化程序来初始化,如下面的代码示例所示:
var orderLineReceiver = new { ReceiverName = "Name Surname", ReceiverAddress = "Some address" };
有关它的更多信息-Object and collection initializers
答案 6 :(得分:0)
对象初始化程序可用于初始化一些小集合,这些小集合可在初始程序创建阶段用于测试。代码示例如下:
class Program
{
static void Main(string[] args)
{
List<OrderLine> ordersLines = new List<OrderLine>()
{
new OrderLine {Platform = "AmazonUK", OrderId = "200-2255555-3000012", ItemTitle = "Test product 1"},
new OrderLine {Platform = "AmazonUK", OrderId = "200-2255555-3000013", ItemTitle = "Test product 2"},
new OrderLine {Platform = "AmazonUK", OrderId = "200-2255555-3000013", ItemTitle = "Test product 3"}
};
}
}
class OrderLine
{
public string Platform { get; set; }
public string OrderId { get; set; }
public string ItemTitle { get; set; }
}
这就是陷阱。在上面的代码示例中,没有包含任何构造函数,并且可以正常工作,但是例如,如果某些带有参数的构造函数将包含在OrderLine类中,例如:
public OrderLine(string platform, string orderId, string itemTitle)
{
Platform = platform;
OrderId = orderId;
ItemTitle = itemTitle;
}
编译器将显示错误-没有给出与所需形式参数相对应的参数。可以通过在OrderLine类中包含不带参数的显式默认构造函数来对其进行修复:
public OrderLine() {}