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

Существует некоторая магия с десериализацией 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? Это техника, которую другие могут использовать, или она скрыта от нас?

Ответ 1

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