XPath与XmlDocument无法找到节点

时间:2016-11-14 01:14:45

标签: c# xml xpath xmldocument

所以我在XPath中没有从XML树中选择任何节点时遇到一些麻烦。到目前为止,这是我的代码:

var reader = new XmlDocument();
reader.Load(@"http://www.fieldgulls.com/rss/current");

XmlNodeList list = reader.SelectNodes("./entry");

我还尝试了* / entry,//条目等的XPath值。我似乎无法获得任何工作。我做错了什么?

2 个答案:

答案 0 :(得分:1)

问题是元素<Entry>实际上位于根节点的默认命名空间中,即"http://www.fieldgulls.com/rss/current"

<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="en"> <!--The default namespace for nested elements is set here with the xmlns= attribute -->
  <title>Field Gulls -  All Posts</title>
  <subtitle>The stupidest name in smart football analysis.</subtitle>
  <icon>https://cdn3.vox-cdn.com/community_logos/50215/fieldgulls-fav.png</icon>
  <updated>2016-11-13T17:00:02-08:00</updated>
  <id>http://www.fieldgulls.com/rss/current/</id>
  <link type="text/html" href="http://www.fieldgulls.com/" rel="alternate"/>
  <entry>
    <!--Remainder commented out-->

因此,您需要使用适当的命名空间和相应的SelectNodes()覆盖选择节点:

var reader = new XmlDocument();
reader.Load(@"http://www.fieldgulls.com/rss/current");

var nsmgr = new XmlNamespaceManager(reader.NameTable);
nsmgr.AddNamespace("a", "http://www.w3.org/2005/Atom");

XmlNodeList list = reader.SelectNodes(".//a:entry", nsmgr);

在这种情况下,我发现使用以下调试实用程序(基于较新的LINQ to XML类库)可以使每个节点的命名空间显而易见:

public static class XObjectExtensions
{
    public static IEnumerable<string> DumpXmlElementNames(this XDocument doc)
    {
        return doc.Root.DumpXmlElementNames();
    }

    public static IEnumerable<string> DumpXmlElementNames(this XElement root)
    {
        if (root == null)
            return Enumerable.Empty<string>();
        var startCount = root.AncestorsAndSelf().Count();
        return root.DescendantsAndSelf().Select(el => string.Format("{0}\"{1}\"",
            new string(' ', 2 * (el.AncestorsAndSelf().Count() - startCount)), el.Name.ToString()));
    }
}

然后,在调试时,你会这样做:

Console.WriteLine("Dumping a list of all element names and namespaces: ");
Console.WriteLine(String.Join("\n", XDocument.Load(@"http://www.fieldgulls.com/rss/current").DumpXmlElementNames()));

产生的输出以:

开头
"{http://www.w3.org/2005/Atom}feed"
  "{http://www.w3.org/2005/Atom}title"
  "{http://www.w3.org/2005/Atom}subtitle"
  "{http://www.w3.org/2005/Atom}icon"
  "{http://www.w3.org/2005/Atom}updated"
  "{http://www.w3.org/2005/Atom}id"
  "{http://www.w3.org/2005/Atom}link"
  "{http://www.w3.org/2005/Atom}entry"

示例fiddle

答案 1 :(得分:0)

尝试使用SyndicationFeed课程。它很容易使用RSS。

using (var xmlReader = XmlReader.Create(@"http://www.fieldgulls.com/rss/current"))
{
    var feed = SyndicationFeed.Load(xmlReader);

    foreach (var item in feed.Items)
    {
        // use item.Title.Text and so on
    }
}