В чем разница между этими двумя вариантами выражений инициализатора коллекции?

Tagc 09/16/2017. 1 answers, 803 views
c# collection-initializer

Я некоторое время использовал C #, но недавно заметил, что поведение одного из моих модульных тестов изменилось в зависимости от того, какую вариацию выражения инициализации коллекции я использовал:

  • var object = new Class { SomeCollection = new List { 1, 2, 3 } };
  • var object = new Class { SomeCollection = { 1, 2, 3 } };

До этого момента я предполагал, что вторая форма была просто синтаксическим сахаром и была семантически эквивалентна первой форме. Однако переход между этими двумя формами привел к тому, что мой провал модульной проверки прошел.

Пример кода ниже показывает это:

 void Main()
{
    var foo1 = new Foo { Items = new List { 1, 2, 3} };
    var foo2 = new Foo { Items = { 1, 2, 3 } };

    foo1.Dump();
    foo2.Dump();
}

class Foo
{
    public List Items { get; set; }
} 

Когда я запускаю это, первое назначение работает нормально, а второе приводит к NullReferenceException .

Я чувствую, что за кулисами компилятор рассматривает эти два выражения так:

 var foo1 = new Foo();
foo1.Items = new List { 1, 2, 3 }; 

var foo2 = new Foo();
foo2.Items.Add(1);
foo2.Items.Add(2);
foo2.Items.Add(3); 

Это предположение верно?

1 Comments
Leon Barkan 07/31/2017
Это поможет вам: c-sharpcorner.com/article/...

1 Answers


Jon Skeet 07/31/2017.

Да, ваше предположение верно. Если инициализатор объекта имеет только:

{
    Property = { ... }
} 

скорее, чем

{
    Property = expression
} 

то setter для свойства не используется - используется getter , а затем вызывается либо метод Add , либо свойства задаются в возвращаемом значении. Так:

var foo = new Foo
{
    Collection = { 1 }
    Property =
    {
        Value = 1
    }
}; 

эквивалентно:

// Only the *getters* for Collection and Property are called
var foo = new Foo();
foo.Collection.Add(1);
foo.Property.Value = 1; 

Сравните это с:

var foo = new Foo
{
    Collection = new List { 1 },
    Property = new Bar { Value = 1 }
}; 

что эквивалентно:

// The setters for Collection and Property are called
var foo = new Foo();
foo.Collection = new List { 1 };
foo.Property = new Bar { Value = 1 }; 
1 comments
1 Ash Burlaczenko 07/31/2017
Не следует ли эквивалент новой коллекции или ее просто добавить?

Related questions

Hot questions

Language

Popular Tags