对象初始值设定项和构造函数之间有什么区别?

时间:2009-04-11 19:24:12

标签: c# .net constructor object-initializer

两者之间有什么区别?何时使用“对象初始值设定项”而不是“构造函数”,反之亦然?我正在使用C#,如果这很重要的话。此外,对象初始化方法是否特定于C#或.NET?

7 个答案:

答案 0 :(得分:220)

对象初始化器是添加到C#3的东西,以便在使用对象时简化对象的构造。

构造函数在给定0个或更多参数的情况下运行,并用于在调用方法获取创建对象的句柄之前创建和初始化对象。例如:

MyObject myObjectInstance = new MyObject(param1, param2);

在这种情况下,MyObject的构造函数将使用值param1param2运行。这些都用于在内存中创建新的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() {}