使用DataContractSerializer在Object中反序列化XML

时间:2014-08-11 20:25:04

标签: c# xml

我有以下xml:

<ArrayOfString>
    <stringValues>
        <stringvalue>teste1</stringvalue>
        <stringList> 
             <stringText>stringList1</stringText>                                   
        </stringList>
    </stringValues>
</ArrayOfString>

我的合同如下:

[KnownType(typeof(StringList))]
[DataContract(Name = "stringValues", Namespace = "")]
public class StringValues
{

    [DataMember(Name="stringvalue",Order=1)]
    public string stringvalue { get; set; }

    [DataMember(Name="stringList",Order=2)]
    public StringList stringList { get; set; }
}

[CollectionDataContract(ItemName = "stringText")]
public class StringList : List<string>
{

}

用于反序列化的代码是:

var testexml = @"<ArrayOfString>
                            <stringValues>
                                <stringvalue>teste1</stringvalue>
                                <stringList> 
                                    <stringText>stringList1</stringText>                                   
                                </stringList>
                            </stringValues>
                         </ArrayOfString>";

        XmlDictionaryReader stringsReader = XmlDictionaryReader.CreateDictionaryReader(XmlReader.Create(new StringReader(testexml)));
        DataContractSerializer ArrayOfString = new DataContractSerializer(typeof(List<StringValues>), "ArrayOfString", "");
        var strings = ArrayOfString.ReadObject(stringsReader);

当我运行它时,stringList属性应该有一个项目,但它没有项目。 我相信,我的合同有问题。有人可以帮忙吗?

提前致谢!

1 个答案:

答案 0 :(得分:1)

调试这样的反序列化问题的最佳方法是创建类的内存示例,然后将其序列化为字符串,并将所得到的内容与要反序列化的内容进行比较。如果存在差异,那就是你的错误。

此处的代码显示了如何为类创建示例序列化XML:

    private static MemoryStream GenerateStreamFromString(string value)
    {
        return new MemoryStream(Encoding.Unicode.GetBytes(value ?? ""));
    }

    private static string GetXml<T>(T obj, DataContractSerializer serializer) where T : class 
    {
        using (var textWriter = new StringWriter())
        {
            XmlWriterSettings settings = new XmlWriterSettings();
            settings.Indent = true;
            settings.IndentChars = "    "; // The indentation used in the test string.
            using (XmlWriter xmlWriter = XmlWriter.Create(textWriter, settings))
            {
                serializer.WriteObject(xmlWriter, obj);
            }
            return textWriter.ToString();
        }
    }

    public static string GetXml<T>(T obj) where T : class
    {
        DataContractSerializer serializer = new DataContractSerializer(typeof(T));
        return GetXml(obj, serializer);
    }

注意 - 不是您问题的完整答案,但代码可能有用且评论时间太长。

<强>更新

好吧,我自己试过了。您需要将上层类的DataContract上的命名空间设置为与嵌套类相同的命名空间,并且与XML文件本身中给出的相同:

[CollectionDataContract(ItemName = "stringText", Namespace="")]
public class StringList : List<string>
{

}

如果你不这样做,DataContractSerializer将假设他们在不同的名称空间中,并期望这样的事情:

<?xml version="1.0" encoding="utf-16"?>
<ArrayOfstringValues xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
    <stringValues>
        <stringvalue>teste1</stringvalue>
        <stringList xmlns:d3p1="http://schemas.datacontract.org/2004/07/Clr.Namespace">
            <d3p1:stringText>stringList1</d3p1:stringText>
        </stringList>
    </stringValues>
</ArrayOfstringValues>

其中Clr.Namespace是您在其中定义类的CLR命名空间。 (是的,显然是DataContractSerializer assigns a default namespace based on the c# namespace,在没有任何指示的情况下另有说明。)

如果您不想重复执行此操作,也可以在整个模块或程序集上使用ContractNamespace属性。