5

我有一个类实际上是一个基于对象的枚举。该类公开了一组静态对象,并且所有内容都使用这些相同的实例。例如(注意私有构造函数)

[DataContract]
public class FieldType
{
    public static readonly FieldType Default  = new FieldType(1, "Default");
    public static readonly FieldType Name     = new FieldType(2, "Name");
    public static readonly FieldType Etc      = new FieldType(3, "Etc");

    private FieldType(uint id, string name)
    {
        Id = id;
        Name = name;
    }

    [DataMember] public uint   Id   { get; private set; }
    [DataMember] public string Name { get; private set; }
    //snip other properties
}

在我必须跨 WCF 序列化之前,这很有效。通过DataContractSerializer绕过构造函数创建新对象。这会产生一个有效的FieldType对象,但它是一个新实例,不是我的静态实例之一。这使得与已知静态值的参考比较失败。

有什么方法可以覆盖类的序列化行为,以便我创建对象实例而不是填充提供给我的实例?

4

2 回答 2

8

怀疑你可以这样做:

[DataContract]
public class FieldType : IObjectReference
{
    object IObjectReference.GetRealObject(StreamingContext ctx)
        switch(Id) {
            case 1: return Default;
            case 2: return Name; // note this is a collision between static/non-static
            case 3: return Etc;
            default: throw new InvalidOperationException();
        }
    }
    public static readonly FieldType Default  = new FieldType(1, "Default");
    // note this is a collision between static/non-static
    public static readonly FieldType Name     = new FieldType(2, "Name");
    public static readonly FieldType Etc      = new FieldType(3, "Etc");

    private FieldType(uint id, string name)
    {
        Id = id;
        Name = name; // note this is a collision between static/non-static
    }

    [DataMember] public uint   Id   { get; private set; }
    // note this is a collision between static/non-static
    [DataMember] public string Name { get; private set; }
    //snip other properties
}

已验证:

public static class Program
{
    static void Main()
    {
        var obj = FieldType.Default;

        using(var ms = new MemoryStream())
        {
            var ser = new DataContractSerializer(typeof (FieldType));
            ser.WriteObject(ms, obj);
            ms.Position = 0;
            var obj2 = ser.ReadObject(ms);

            bool pass = ReferenceEquals(obj, obj2); // true
        }
    }
}

但是请注意,Name如果我们只使用Id来识别要使用的真实对象,那么序列化似乎没有什么意义。

于 2012-04-11T11:06:32.490 回答
5

我建议您覆盖Equalsand GetHashcode(和==and !=),以便您对静态对象与 WCF 创建的对象进行相等性检查。

数据传输对象 (DTO) 不适用于面向对象的行为,它们是纯粹的状态类。但我能理解你面临的问题。

或者,在您的域对象与上述类一起使用时,使用不同的 DTO 发送数据。

于 2012-04-11T11:06:27.030 回答