【问题标题】:Deserializing XML returns null for collection property反序列化 XML 为集合属性返回 null
【发布时间】:2013-02-08 19:32:46
【问题描述】:

我正在尝试反序列化以下 XML:

<?xml version="1.0" encoding="utf-8" ?> 
<mf:somedata xmlns:mf="urn:somedata">
    <CurrentAccount>
        <AccountType>test</AccountType>
        <Charge>
            <ChargeType>test</ChargeType>
        </Charge>
    </CurrentAccount>
    <CurrentAccount>
        <AccountType>test 2</AccountType>
        <Charge>
            <ChargeType>test 2</ChargeType>
        </Charge>
    </CurrentAccount>
</mf:somedata>

使用以下类:

[XmlRoot("somedata", Namespace = "urn:somedata")]
public class MfCurrentAccounts
{
    [XmlElement("CurrentAccount")]
    public CurrentAccount[] CurrentAccounts { get; set; }
}

public class CurrentAccount
{
    public string AccountType { get; set; }

    [XmlElement("Charge")]
    public Charge[] Charges { get; set; }
}

public class Charge
{
    public string ChargeType { get; set; }
}

var c = new XmlSerializer(typeof(MfCurrentAccounts)).Deserialize(new StringReader(xml)) as MfCurrentAccounts;

c.CurrentAccounts // <-- is always null

但无论我尝试什么,当我反序列化 CurrentAccounts 数组时它都是空的。我已经尝试了所有我能想到的属性组合(我也尝试过 XmlArray 和 XmlArrayItem)。

我做错了什么? :S

【问题讨论】:

  • 对不起,用反序列化代码更新了问题。
  • 你不需要用[Serializable()]标记类吗?
  • @François 我试过了,但显然不需要。
  • @Dan-o 当 this 代码已经按照该答案的建议执行时,这怎么可能是重复的? this 代码的注释/属性与“重复答案”的唯一区别是它使用数组而不是列表。

标签: c# xml deserialization xmlserializer


【解决方案1】:

问题在于命名空间。

当我在测试情况下创建整个类设置时,我得到了一个非常不同的输出。这是我认为您应该尝试阅读的内容:

    <?xml version="1.0"?>
    <mf:somedata xmlns:mf="urn:somedata">
        <mf:CurrentAccount>
            <mf:AccountType>something 1</mf:AccountType>
            <mf:Charge>
                <mf:ChargeType>Charge Type 1</mf:ChargeType>
            </mf:Charge>
        </mf:CurrentAccount>
        <mf:CurrentAccount>
            <mf:AccountType>something 2</mf:AccountType>
            <mf:Charge>
                <mf:ChargeType>Charge Type 2</mf:ChargeType>
            </mf:Charge>
        </mf:CurrentAccount>
    </mf:somedata>

注意所有额外的mf:。当您声明命名空间时,序列化程序将使用该命名空间,并且仅反序列化正确属于该命名空间的节点。您要么需要完全摆脱它,要么适当地修复您的输入。这是我用来生成该输出的代码注意:类定义完全没有改变

XmlSerializer ser = new XmlSerializer(typeof(MfCurrentAccounts));
XmlSerializerNamespaces ns = new XmlSerializerNamespaces();
ns.Add("mf", "urn:somedata");

MemoryStream ms = new MemoryStream();
ser.Serialize(ms, a, ns);

当重新阅读时:

ms.Position = 0;
b = ser.Deserialize(ms) as MfCurrentAccounts;

在运行这两个部分之后,b 现在是 a 的完美克隆,我上面显示的 xml 就是生成的 xml。

【讨论】:

  • 谢谢。不幸的是,XML 来自第三方,所以我无法控制它。我尝试将此添加到 CurrentAccounts 属性:[XmlElement("CurrentAccount", Namespace = "urn:somedata")] 但无济于事。
  • 在尝试反序列化之前可以编辑 xml 吗?如果您在第一行使用一些正则表达式并从顶部节点中删除“mf:”,那么您将拥有一个统一的 classdef,它应该正确反序列化......
  • @Tom 你试过 [XmlElement("CurrentAccount", Namespace = String.Empty)] 吗?
  • @jbl 你是救生员!谢谢! :)
  • @jbl 你救了一条命 Namespace = "",工作!!
【解决方案2】:

也许您应该尝试将您的 MfCurrentAccounts 类替换为:

[XmlRoot("somedata", Namespace = "urn:somedata")]
public class MfCurrentAccounts : List<CurrentAccount>
{
   public MfCurrentAccounts():base()
   {}

}

或者看看https://stackoverflow.com/a/364410/1236044

希望这会有所帮助

【讨论】:

  • 这只是返回一个空列表。 :( 不过我会玩弄它,因为它似乎应该可以工作。我会看看写入磁盘并使用 XSD 文件,但我不想走那条路。
  • @Tom 这个类可能需要一个公共构造函数。我更新了我的答案
  • 这似乎也不起作用,但您对先前答案的评论有效。 :)
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-05-14
相关资源
最近更新 更多