我有这种情况:
[System.CodeDom.Compiler.GeneratedCodeAttribute("xsd", "4.0.30319.33440")]
[System.SerializableAttribute()]
[System.Diagnostics.DebuggerStepThroughAttribute()]
[System.ComponentModel.DesignerCategoryAttribute("code")]
[System.Xml.Serialization.XmlTypeAttribute(AnonymousType = true)]
[System.Xml.Serialization.XmlRootAttribute(Namespace = "", IsNullable = false)]
public class Response
{
public string RetKey { get; set; }
public string RetType { get; set; }
}
[Test]
public void Test3()
{
{
var s = @"<Response CommandID=""57b48ae28f484ab4b3cc2e841cddd02a"">
<RetKey>5</RetKey>
<RetType>152</RetType>
</Response>";
var xmlSerializer = new XmlSerializer(typeof(Response));
var deserialize = (Response)xmlSerializer.Deserialize(new StringReader(s));
Assert.AreEqual("5", deserialize.RetKey, "no namespace");
}
{
var s = @"<Response xmlns=""http://www.sap.com/SBO/DIS"" CommandID=""57b48ae28f484ab4b3cc2e841cddd02a"">
<RetKey>5</RetKey>
<RetType>152</RetType>
</Response>";
var xRoot = new XmlRootAttribute
{
ElementName = typeof(Response).Name,
Namespace = "http://www.sap.com/SBO/DIS",
IsNullable = true
};
var xmlSerializer = new XmlSerializer(typeof(Response), xRoot);
var deserialize = (Response)xmlSerializer.Deserialize(new StringReader(s));
Assert.AreEqual("5", deserialize.RetKey, "try to declare namespaces");
}
}
第二次尝试失败(使用命名空间)。 因此,我陷入了XML反序列化中正确使用命名空间的困境。 令我感到困惑的是,反序列化过程没有错误,但是返回了空对象。
我缺少什么?
答案 0 :(得分:5)
第二个版本的xml 完全不同,因此它失败是正确的。命名空间是xml节点身份的基本。如果要使用第二个版本,则需要告诉它所有相关Namespace="http://www.sap.com/SBO/DIS"
属性中的[Xml...]
,并注意xmlns="..."
由子元素继承。这意味着您还需要告诉它RetKey
和RetType
在该命名空间中。
如果要在运行时定义它,则:
var xRoot = new XmlRootAttribute
{
ElementName = nameof(Response),
Namespace = "http://www.sap.com/SBO/DIS",
IsNullable = true
};
var xor = new XmlAttributeOverrides();
xor.Add(typeof(Response), nameof(Response.RetKey), new XmlAttributes
{
XmlElements = { new XmlElementAttribute(nameof(Response.RetKey))
{ Namespace = xRoot.Namespace } }
});
xor.Add(typeof(Response), nameof(Response.RetType), new XmlAttributes
{
XmlElements = { new XmlElementAttribute(nameof(Response.RetType))
{ Namespace = xRoot.Namespace } }
});
var xmlSerializer = new XmlSerializer(typeof(Response),
xor, Type.EmptyTypes, xRoot, xRoot.Namespace);
但是,请注意,以这种方式创建XmlSerializer
时,您必须缓存并重新使用它-每当您new XmlSerializer(...)
时,它都会生成新的程序集这样,它很快就会变得非常昂贵。所以:将其存放在某个地方!或者只是使用属性(使用简单的构造方法时,它会在内部缓存)。