3

我试图序列化类B作为它的基类A的实例。 DataContractSerializer不会允许我这样做。将继承类型的DataContract序列化为基类型

一个例子失败的系列化如下:

class Program 
{ 
    [DataContract] 
    public class A 
    { 
     public int Id { get; set; } 
    } 

    [DataContract] 
    public class B : A 
    { 

    } 


    static void Main(string[] args) 
    { 
     A instance = new B { Id = 42 }; 

     var dataContractSerializer = new DataContractSerializer(typeof(A)); 
     var xmlOutput = new StringBuilder(); 
     using (var writer = XmlWriter.Create(xmlOutput)) 
     { 
      dataContractSerializer.WriteObject(writer, instance); 
     } 

    } 
} 

我知道,这个问题很容易解决通过添加KnownTypes属性。 但是我想保留从项目中隐藏的类B(不添加引用)。

完全可以实现我想要的吗?我尝试了XmlSerializer,但它给了我同样的问题(它在XML中添加了完整的实例类型名称),并且使用起来更笨拙。

+0

它必须是XML?或者必须只是工作? – 2010-12-10 16:15:12

+0

它可以工作。尽管最小麻烦会很好。 – R4cOON 2010-12-13 07:37:37

回答

3
[DataContract(Name="YouCannotSeeMyName")] 
[KnownTypes(typeof(B))] 
public class B : A 

,你会得到

<A itype="YouCannotSeeMyName"> 
    ... 
</A> 
+0

我认为这仍然会序列化B的所有成员,这不是OP的意图。 – 2010-12-10 15:50:22

+0

@Mark Avenius:如果OP没有在B的属性上放置'[DataMember]'属性,那么不需要。 – 2010-12-10 15:57:25

+0

优秀点:-) – 2010-12-10 15:59:12

1

我敢肯定,你不能隐藏合同的部分。这类似于处理一个Web服务,在这个Web服务中,为了理解每个端点如何以及如何进行序列化/反序列化,合同必须得到尊重。

另外,您可以在DataContractSerializer范围内传递B类型,而不使用该属性。

class Program 
    { 
     [DataContract] 
     public class A 
     { 
      public int Id { get; set; } 
     } 

     [DataContract] 
     public class B : A 
     { 

     } 

     static void Main(string[] args) 
     { 
      A instance = new B { Id = 42 }; 

      var dataContractSerializer = new DataContractSerializer(typeof(A), new List<Type>() { typeof(B) }); 
      var xmlOutput = new StringBuilder(); 
      using (var writer = XmlWriter.Create(xmlOutput)) 
      { 
       dataContractSerializer.WriteObject(writer, instance); 
      } 

     } 
    } 

,这将给你...

<Program.A xmlns:i="http://www.w3.org/2001/XMLSchema-instance" i:type="Program.B" 
xmlns="http://schemas.datacontract.org/2004/07/ConsoleApplication1" />