LINQ XML搜索失败,默认属性路径

时间:2013-05-22 10:01:20

标签: c# xml linq

我使用以下代码搜索元素数量,并且只有在没有默认路径时才搜索搜索成功:

要搜索的代码:

XElement root = XElement.Load(@"c:\b.txt", LoadOptions.PreserveWhitespace);
IEnumerable<XElement> address =
             from el in root.Elements("Address")
              select el;
int c = address.Count();

c的值为2,其中包含以下数据:

<?xml version="1.0" encoding="UTF-8"?>
<presence xmlns:a="urn:ietf:params:xml:ns:pidf"
  xmlns:dm="urn:ietf:params:xml:ns:pidf:data-model"
  xmlns:oma="urn:xml:prs:pidf:oma-pres"
  entity="sip:john@police.city.gov">
  <Address Type="Shipping">
    <Name>Ellen Adams</Name>
    <Street>123 Maple Street</Street>
    <City>Mill Valley</City>
    <State>CA</State>
    <Zip>10999</Zip>
    <Country>USA</Country>
  </Address>
  <Address Type="Billing">
    <Name>Tai Yee</Name>
    <Street>8 Oak Avenue</Street>
    <City>Old Town</City>
    <State>PA</State>
    <Zip>95819</Zip>
    <Country>USA</Country>
  </Address>
</presence>

但如果我通过将第二行换成(xmlns而不是xmlns:a)来更改XML:

 <presence xmlns="urn:ietf:params:xml:ns:pidf"

我得到的值0不正确。

有什么建议吗?

由于

2 个答案:

答案 0 :(得分:4)

xmlns="urn:ietf:params:xml:ns:pidf"表示您为xml中没有指定任何名称空间的所有元素设置默认名称空间。

结果,您还应该将命名空间声明添加到LINQ to XML查询中,如下所示:

XElement root = XElement.Load(@"c:\b.txt", LoadOptions.PreserveWhitespace);

XNamespace xmlns = "urn:ietf:params:xml:ns:pidf";

IEnumerable<XElement> address = root.Elements(xmlns + "Address");

Console.WriteLine(address.Count()); //prints 2

或者您可以使用命名空间无关的方法,无论指定的默认命名空间是什么,它都将起作用:

var address = root.Elements()
                  .Where(node => node.Name.LocalName == "Address");
//address will contain the same nodes, as in previous example

另请注意,在这种情况下,扩展方法语法更清晰。

答案 1 :(得分:4)

通过设置xmlns =“urn:ietf:params:xml:ns:pidf”,您已为所有元素设置了默认命名空间。因此,“Address”元素不再存在。它现在被称为“urn:ietf:params:xml:ns:pidf:Address”。

您需要做的是声明XNamespace并将其添加到元素名称:

XNamespace defaultNamespace = "urn:ietf:params:xml:ns:pidf";
XElement root = XElement.Load(@"c:\b.txt", LoadOptions.PreserveWhitespace);
IEnumerable<XElement> address =
             from el in root.Elements(defaultNamespace + "Address")
              select el;
int c = address.Count();