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您会看到字段_wasConstructorCalledfalse

那么,WCF如何做到这一点?这是其他人也可以使用的技术,还是对我们隐藏?

Answers:


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
好吧,我以前发布了错误的答案(现已删除),因此感到内gui。没有什么比挫伤程序员自我让他去做一些研究了。
杰森·杰克逊

3
现在是否还有其他人想知道,那么FormatterServices.GetUninitializedObject如何工作?反射?
harpo 2010年

如果我还记得它调用本机代码的话。我再也不能用Reflector在兔子洞下走了。
杰森·杰克逊

6
很奇怪-我在linqpad中运行该代码,然后得到:0 0作为输出。其实这对我来说很有意义,因为字段初始被联到构建函数AFAIK
用套管

1
@bushed是正确的。我在此处发布了包含代码和结果的屏幕截图。起初我以为.NET框架版本可能有所不同(因为答案已经有4年的历史了),但我检查了2.0和4.0,它们都向控制台写入了0和0。杰森·杰克逊(Jason Jackson),您能否更新您的帖子以反映这些发现?
奥利弗(Oliver)

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.