4

我面临的情况是使用 Linq 将 XML 文档解析为对象。在解析过程中,我正在检查以确保元素不为空,然后再继续解析它们的值。有没有办法简化这个陈述?

  var variable = (from x in xdoc.Descendants("Root")
                 select new AccountingResponse
                 {      
                 NetCharge = x.Element("Charges") != null && x.Element("Charges").Element("NetCharge") != null ? x.Element("Charges").Element("NetCharge").Value : "0",
                 TotalCharge = x.Element("Charges") != null && x.Element("Charges").Element("TotalCharge") != null ? x.Element("Charges").Element("TotalCharge").Value : "0"
                 }).SingleOrDefault();

总而言之,我不想继续检查每条线上是否存在节点。我知道我可以在解析之前测试该节点是否存在,但可能还有其他数据需要解析以创建 AccountingResponse,我想避免一次只解析一部分 XML 的 if 语句。

或者也许我做错了,有更好的方法!

4

3 回答 3

7

一个简单的选项是使用Elements而不是Element- 如果元素不存在,它将返回零长度序列。所以你可以使用:

from x in xdoc.Descendants("Root")
select new AccountingResponse
{      
    NetCharge = x.Elements("Charges")
                 .Elements("NetCharge")
                 .Select(y => (int) y)
                 .FirstOrDefault(),
    TotalCharge = x.Elements("Charges")
                   .Elements("TotalCharge")
                   .Select(y => (int) y)
                   .FirstOrDefault(),
}).SingleOrDefault();

(请注意,您的原始代码不会编译,因为Value它是一个字符串,而 0 是一个 int ...)

于 2013-04-05T15:33:11.650 回答
3

在 C# 6.0 中,您可以使用 monadic Null-conditional operator?. 在您的示例中应用它之后,它看起来像这样:

var variable = (from x in xdoc.Descendants("Root")
                select new
                {
                    NetCharge = x.Element("Charges")?.Element("NetCharge")?.Value ?? "0",
                    TotalCharge = x.Element("Charges")?.Element("TotalCharge")?.Value ?? "0"
                }).SingleOrDefault();

你可以在这里阅读更多标题为 Null 条件运算符的内容。

于 2015-09-06T09:40:31.923 回答
2

你可以更换

x.Element("nodeName") != null : int.Parse(x.Element("nodeName").Value) : 0

(int)x.Element("nodeName")

它也适用于string, int, double, decimal, bool, uint,DateTimeNullable那些。

于 2013-04-05T15:32:52.853 回答