1. ホーム
  2. c#

WCFのデシリアライズは、コンストラクタを呼び出すことなく、どのようにオブジェクトをインスタンス化するのですか?

2023-08-31 17:11:25

質問

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 というフィールドがあることがわかるでしょう。 _wasConstructorCalledfalse .

では、WCFはどのようにこれを行うのでしょうか? これは他の人も使えるテクニックなのでしょうか、それとも私たちから隠されているのでしょうか?

どのように解決するのか?

FormatterServices.GetUninitializedObject() はコンストラクタを呼び出さずにインスタンスを生成します。 このクラスを見つけるには リフレクタ を使い、.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