Как десериализация WCF создает экземпляры объектов без вызова конструктора?

79

С десериализацией WCF творится некое волшебство. Как он создает экземпляр типа контракта данных без вызова его конструктора?

Например, рассмотрим этот контракт данных:

[DataContract]
public sealed class CreateMe
{
   [DataMember] private readonly string _name;
   [DataMember] private readonly int _age;
   private readonly bool _wasConstructorCalled;

   public CreateMe()
   {
      _wasConstructorCalled = true;
   }

   // ... other members here
}

При получении экземпляра этого объекта через DataContractSerializerвы увидите, что поле _wasConstructorCalledесть false.

Итак, как WCF это делает? Может ли эта техника использоваться и другими, или она скрыта от нас?

Дрю Ноукс
источник

Ответы:

102

FormatterServices.GetUninitializedObject()создаст экземпляр без вызова конструктора. Я нашел этот класс, используя Reflector и покопавшись в некоторых основных классах сериализации .Net.

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

using System;
using System.Reflection;
using System.Runtime.Serialization;

namespace NoConstructorThingy
{
    class Program
    {
        static void Main()
        {
            // does not call ctor
            var myClass = (MyClass)FormatterServices.GetUninitializedObject(typeof(MyClass));

            Console.WriteLine(myClass.One); // writes "0", constructor not called
            Console.WriteLine(myClass.Two); // writes "0", field initializer not called
        }
    }

    public class MyClass
    {
        public MyClass()
        {
            Console.WriteLine("MyClass ctor called.");
            One = 1;
        }

        public int One { get; private set; }
        public readonly int Two = 2;
    }
}

http://d3j5vwomefv46c.cloudfront.net/photos/large/687556261.png

Джейсон Джексон
источник
6
Ну, я раньше публиковал неправильный ответ (теперь удален), поэтому я чувствовал себя виноватым. Нет ничего лучше, чем ушибить эго программиста, чтобы заставить его провести какое-то исследование.
Джейсон Джексон,
3
Кто-нибудь еще задается вопросом, как же тогда работает FormatterServices.GetUninitializedObject? Отражение?
harpo
Если я помню, это вызов в собственный код. Я не мог следить за этим дальше по кроличьей норе с помощью Reflector.
Джейсон Джексон
6
Странно - я запустил этот код в linqpad и получил: 0 0 на выходе. На самом деле для меня это имеет смысл, поскольку инициализаторы полей встроены в ctors AFAIK
bushed
1
@bushed правильный. Я разместил скриншот с кодом и привести сюда . Сначала я подумал, что это может быть разница в версиях .NET framework (поскольку ответу уже 4 года), но я проверил 2.0 и 4.0, и они оба пишут 0 и 0 в консоль. Джейсон Джексон, не могли бы вы обновить свой пост, чтобы отразить эти результаты?
Оливер