2017-02-21 9 views
3

は、私は次のクラスにXMLをデシリアライズしたい:XmlSerializerで特定の型のすべてのメンバーを無視する方法を教えてください。

public partial class Delivery 
{ 
    public System.Nullable<System.DateTime> sentDate { get; set; } 
    public System.Nullable<System.DateTime> receivedDate { get; set; } 
    public System.Nullable<System.DateTime> responseDueDate { get; set; } 
} 

しかし、XMLの日付は、XmlSerializerを友好的なフォーマットではありません。

Message=The string '2010-06-12T00:00:00 -05:00' is not a valid AllXsd value. 
Source=System.Xml.ReaderWriter 
StackTrace: 
    at System.Xml.Schema.XsdDateTime..ctor(String text, XsdDateTimeFlags kinds) 
    at System.Xml.XmlConvert.ToDateTime(String s, XmlDateTimeSerializationMode dateTimeOption) 
    at Microsoft.Xml.Serialization.GeneratedAssembly.XmlSerializationReaderDeserializedAudit.Read1_NullableOfDateTime(Boolean checkType) 
    at Microsoft.Xml.Serialization.GeneratedAssembly.XmlSerializationReaderDeserializedAudit.Read15_DateSafeDelivery(Boolean isNullable, Boolean checkType) 
    at Microsoft.Xml.Serialization.GeneratedAssembly.XmlSerializationReaderDeserializedAudit.Read16_MyParent(Boolean isNullable, Boolean checkType) 

だからDateSafeDeliveryがされている:これは、次のexpectionになり

private static XmlAttributeOverrides GetOverrides() 
{ 
    var overrides = new XmlAttributeOverrides(); 
    var attributes = new XmlAttributes(); 
    attributes.XmlElements.Add(new XmlElementAttribute(typeof(DateSafeDelivery))); 
    overrides.Add(typeof(MyParent), "Delivery", attributes); 
    var ignore = new XmlAttributes { XmlIgnore = true }; 
    overrides.Add(typeof(DateTime?), ignore); 
    return overrides; 
} 

:私はその後、私のオーバーライドを設定

public partial class DateSafeDelivery : Delivery 
{ 
    [XmlElement("sentDate")] 
    public string sentDateString 
    { 
     internal get { return sentDate.HasValue ? XmlConvert.ToString(sentDate.Value) : null; } 
     set { sentDate = DateTime.Parse(value); } 
    } 
    [XmlElement("receivedDate")] 
    public string receivedDateString 
    { 
     internal get { return receivedDate.HasValue ? XmlConvert.ToString(receivedDate.Value) : null; } 
     set { receivedDate = DateTime.Parse(value); } 
    } 
    [XmlElement("responseDueDate")] 
    public string responseDueDateString 
    { 
     internal get { return responseDueDate.HasValue ? XmlConvert.ToString(responseDueDate.Value) : null; } 
     set { responseDueDate = DateTime.Parse(value); } 
    } 
} 

:mulitple質問への回答に基づいて、私はこのクラスを追加しました日付のXmlIgnoreは無視されています。

overrides.Add(typeof(DateTime?), ignore); 

で:

new Dictionary<string, Type>() 
    { 
     { "sentDate", typeof(Delivery) }, 
     { "receivedDate", typeof(Delivery) }, 
     { "responseDueDate", typeof(Delivery) }, 
    } 
     .ToList() 
     .ForEach(t1 => overrides.Add(t1.Value, t1.Key, ignore)); 

をそして、それは一つのクラスと3つのプロパティの罰金だ

それは私が切り替えた場合、動作します。しかし、私は合計で3ダースの日付のプロパティを持つ14のクラスを持っています。 14クラスのオーバーライドを追加する必要があることはわかっていますが、シリアライザですべてのDateTimeプロパティを無視する方法がありますか?

私はXmlAttributeOverrides.Add Method (Type, XmlAttributes)と思っていました。しかし、それは動作していません。どうして?この方法は何ですか?それは何をするためのものか?

答えて

1

XmlAttributeOverrides.Add(Type, XmlAttributes)は、その型の値を返すすべてのプロパティではなく、XMLオーバーライド属性を型自体に追加するように設計されています。例えば。あなたはDateSafeDelivery[XmlRoot("OverrideName")]属性を追加したい場合は、あなたが何かを行うことができます:

overrides.Add(typeof(DateSafeDelivery), 
    new XmlAttributes { XmlRoot = new XmlRootAttribute("OverrideName") }); 

のすべてのプロパティの直列化を抑制することができる何static XML serialization attributeがないため、指定された型を返すすべてのプロパティを無視する動的なoverride属性はありません特定のタイプの[XmlIgnore]のみプロパティまたはフィールドに適用することができますので、以下でも、コンパイルされません。

[XmlIgnore] public class IgnoreAllInstancesOfMe { } // Fails to compile. 

を(マイクロソフトが[XmlIgnore]のサポートを実装しなかった理由についてはタイプに適用される - あなたはそれらを依頼する必要があります。)

したがって次のような拡張メソッドを導入する必要があります。

public static partial class XmlAttributeOverridesExtensions 
{ 
    public static XmlAttributeOverrides IgnorePropertiesOfType(this XmlAttributeOverrides overrides, Type declaringType, Type propertyType) 
    { 
     return overrides.IgnorePropertiesOfType(declaringType, propertyType, new HashSet<Type>()); 
    } 

    public static XmlAttributeOverrides IgnorePropertiesOfType(this XmlAttributeOverrides overrides, Type declaringType, Type propertyType, HashSet<Type> completedTypes) 
    { 
     if (overrides == null || declaringType == null || propertyType == null || completedTypes == null) 
      throw new ArgumentNullException(); 
     XmlAttributes attributes = null; 
     for (; declaringType != null && declaringType != typeof(object); declaringType = declaringType.BaseType) 
     { 
      // Avoid duplicate overrides. 
      if (!completedTypes.Add(declaringType)) 
       break; 
      foreach (var property in declaringType.GetProperties(BindingFlags.Public | BindingFlags.DeclaredOnly | BindingFlags.Instance)) 
      { 
       if (property.PropertyType == propertyType || Nullable.GetUnderlyingType(property.PropertyType) == propertyType) 
       { 
        attributes = attributes ?? new XmlAttributes { XmlIgnore = true }; 
        overrides.Add(declaringType, property.Name, attributes); 
       } 
      } 
     } 
     return overrides; 
    } 
} 

をと実行します。

private static XmlAttributeOverrides GetOverrides() 
    { 
     var overrides = new XmlAttributeOverrides(); 

     var attributes = new XmlAttributes(); 
     attributes.XmlElements.Add(new XmlElementAttribute(typeof(DateSafeDelivery))); 
     overrides.Add(typeof(MyParent), "Delivery", attributes); 

     // Ignore all DateTime properties in DateSafeDelivery 
     var completed = new HashSet<Type>(); 
     overrides.IgnorePropertiesOfType(typeof(DateSafeDelivery), typeof(DateTime), completed); 
     // Add the other 14 types as required 

     return overrides; 
    } 

DateSafeDeliveryDateString性質が完全に公開されていないプロパティをシリアル化することはできません公共 getおよびsetメソッド、例えば:

public partial class DateSafeDelivery : Delivery 
{ 
    [XmlElement("sentDate")] 
    public string sentDateString 
    { 
     get { return sentDate.HasValue ? XmlConvert.ToString(sentDate.Value, XmlDateTimeSerializationMode.Utc) : null; } 
     set { sentDate = DateTime.Parse(value); } 
    } 

XmlSerializerを持たなければならないことに注意してください。

ところで、this answerで説明したように、静的に、深刻なメモリリークを避けるために、オーバーライドで構成いかなるXmlSerializerをキャッシュする必要がありますのでご注意ください。

+1

XmlAttributeOverrides.Add(Type、XmlAttributes)は、逆シリアル化される型のみです。とった。 最後に、コードの各プロパティをリストしました。私は反射を考慮しましたが、コンピュータに3分かかると毎回把握したくなかったのです。 また、私はメモリリークの回答を見て、静的にXmlSerializerを使用しています。 そして、内部取得はデシリアライズのために機能しています。それが私がこのクラスで気にしている唯一の方向です。 ありがとうございます。 –

関連する問題