193

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

4

7 回答 7

255

对象初始化器是添加到 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;

但是,在多线程环境中,对象初始化程序的原子性可能是有益的,因为它可以防止对象处于未完全初始化的状态(有关更多详细信息,请参阅此答案) - 它可以为 null 或按您的意图进行初始化。

此外,对象初始化器更易于阅读(尤其是当您设置多个值时),因此它们为您提供与构造函数上的许多重载相同的好处,而无需使该类的 API 复杂化的许多重载。

于 2009-04-11T19:42:34.897 回答
52

构造函数是在类型上定义的方法,它接受指定数量的参数,用于创建和初始化对象。

对象初始化程序是在构造函数之后在对象上运行的代码,可用于简洁地将对象上的任意数量的字段设置为指定值。这些字段的设置发生调用构造函数之后。

如果构造函数充分设置了对象的初始状态,则可以在没有对象初始值设定项的帮助下使用构造函数。然而,对象初始值设定项必须与构造函数一起使用。该语法要求显式或隐式使用(VB.Net 和 C#)构造函数来创建初始对象。当构造函数没有充分初始化对象以供您使用并且一些简单的字段和/或属性集将使用时,您将使用对象初始化器。

于 2009-04-11T19:27:20.490 回答
35

当你这样做

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;

现在这促进了这样行为。了解对象初始化器的工作原理很重要。

于 2013-05-31T06:04:24.530 回答
17

如果您必须在对象上设置属性才能使其正常工作,一种方法是只公开一个需要这些强制属性作为参数的构造函数。

在这种情况下,如果不指定这些必需属性,您将无法创建对象。对象初始化器无法强制执行类似的操作。

对象初始化器实际上只是缩短初始分配的“语法便利”。不错,但在功能上并不是很相关。

马克

于 2009-04-11T19:33:40.973 回答
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) }

如您所见,对象初始化器允许您在执行构造的同时为公共字段和公共(可设置)属性指定值,这在构造函数不提供任何重载初始化某些字段时特别有用。但是请注意,对象初始化器只是语法糖,编译后与一系列赋值并没有真正的区别。

于 2009-04-11T19:32:50.927 回答
1

对象初始化器可用于初始化一些小集合,这些集合可用于在初始程序创建阶段进行测试。代码示例如下:

    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() {}
于 2019-10-10T09:39:22.790 回答
0

对象初始值设定项在 LINQ 查询表达式中特别有用。查询表达式经常使用匿名类型,只能使用对象初始化器来初始化,如下面的代码示例所示:`

var orderLineReceiver = new { ReceiverName = "Name Surname", ReceiverAddress = "Some address" };

更多关于它 -对象和集合初始化器

于 2019-10-10T09:01:17.340 回答