web-dev-qa-db-ja.com

XmlSerializerで逆シリアル化の名前空間を無視できますか?

XmlSerializerに、逆シリアル化時に名前空間(xmlns属性)を無視させて、属性が追加されているかどうか、または属性が偽であっても問題にならないようにできますか?ソースは常に信頼されるため、xmlns属性は気にしません。

39
NotDan

はい、XmlSerializerに逆シリアル化中に名前空間を無視するように指示できます。

名前空間を無視するXmlTextReaderを定義します。そのようです:

// helper class to ignore namespaces when de-serializing
public class NamespaceIgnorantXmlTextReader : XmlTextReader
{
    public NamespaceIgnorantXmlTextReader(System.IO.TextReader reader): base(reader) { }

    public override string NamespaceURI
    {
        get { return ""; }
    }
}

// helper class to omit XML decl at start of document when serializing
public class XTWFND  : XmlTextWriter {
    public XTWFND (System.IO.TextWriter w) : base(w) { Formatting= System.Xml.Formatting.Indented;}
    public override void WriteStartDocument () { }
}

TextReaderを使用して逆シリアル化する方法の例を次に示します。

public class MyType1 
{
    public string Label
    {
        set {  _Label= value; } 
        get { return _Label; } 
    }

    private int _Epoch;
    public int Epoch
    {
        set {  _Epoch= value; } 
        get { return _Epoch; } 
    }        
}



    String RawXml_WithNamespaces = @"
      <MyType1 xmlns='urn:booboo-dee-doo'>
        <Label>This document has namespaces on its elements</Label>
        <Epoch xmlns='urn:aaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'>0</Epoch>
      </MyType1>";


    System.IO.StringReader sr;
    sr= new System.IO.StringReader(RawXml_WithNamespaces);
    var o1= (MyType1) s1.Deserialize(new NamespaceIgnorantXmlTextReader(sr));
    System.Console.WriteLine("\n\nDe-serialized, then serialized again:\n");
    s1.Serialize(new XTWFND(System.Console.Out), o1, ns);
    Console.WriteLine("\n\n");

結果は次のようになります。

    <MyType1>
      <Label>This document has namespaces on its elements</Label>
      <Epoch>0</Epoch>
    </MyType1>
44
Cheeso

名前空間はないが、入力に名前空間がある場合は、設定できます

名前空間= false

xmlTextReaderで。

23

ヴォルフガング・グリンフェルドの例外的な回答(例外処理なし):

public static Message Convert(XmlDocument doc)
{
    Message obj;
    using (TextReader textReader = new StringReader(doc.OuterXml))
    {
        using (XmlTextReader reader = new XmlTextReader(textReader))
        {
            reader.Namespaces = false;
            XmlSerializer serializer = new XmlSerializer(typeof(Message));
            obj = (Message)serializer.Deserialize(reader);
        }
    }

    return obj;
}
11
klm_

XmlSerializer Deserializeを使用してストリームからではなくxmlから読み取ることでこれを解決しました。正規表現を使用してxmlからxsi:typeを削除し、xmlが逆シリアル化される前のこの方法。これを行っていたのは、クロスプラットフォーム用のポータブルクラスライブラリなので、他の多くのオプションはありませんでした。

次のコードが役立ちます、

public static TClass Deserialize<TClass>(string xml) where TClass : class, new()
{
    var tClass = new TClass();

    xml = RemoveTypeTagFromXml(xml);

    var xmlSerializer = new XmlSerializer(typeof(TClass));
    using (TextReader textReader = new StringReader(xml))
    {
        tClass = (TClass)xmlSerializer.Deserialize(textReader);
    }
    return tClass;
}

public static string RemoveTypeTagFromXml(string xml)
{
    if (!string.IsNullOrEmpty(xml) && xml.Contains("xsi:type"))
    {
        xml = Regex.Replace(xml, @"\s+xsi:type=""\w+""", "");
    }
    return xml;
}
0

これは名前空間を無視しませんが、代わりにそれを期待します。私はあなたと同じことをしようとしていましたが、XSDを使用した検証に追加して以来、名前空間が必要になりました。だから、ここに私が名前空間を期待していたものがあります。 https://stackoverflow.com/a/7730989/1856992

0
thinklarge