как использовать XPath с XDocument?

95

Есть аналогичный вопрос, но кажется, что решение не получилось в моем случае: Weirdness с XDocument, XPath и пространствами имен

Вот XML, с которым я работаю:

<?xml version="1.0" encoding="utf-8"?>
<Report Id="ID1" Type="Demo Report" Created="2011-01-01T01:01:01+11:00" Culture="en" xmlns="http://demo.com/2011/demo-schema">
    <ReportInfo>
        <Name>Demo Report</Name>
        <CreatedBy>Unit Test</CreatedBy>
    </ReportInfo>
</Report>

И ниже приведен код, который я думал, что он должен работать, но он не...

XDocument xdoc = XDocument.Load(@"C:\SampleXML.xml");
XmlNamespaceManager xnm = new XmlNamespaceManager(new NameTable()); 
xnm.AddNamespace(String.Empty, "http://demo.com/2011/demo-schema");
Console.WriteLine(xdoc.XPathSelectElement("/Report/ReportInfo/Name", xnm) == null);

Есть ли у кого-нибудь идеи? Спасибо.

  • 0
    Посмотрите другой ответ ниже, он не работает, так как реализация XPath 1.0 не может справиться с пустым префиксом.
  • 1
    Как уже было сказано, не используйте пустой префикс при добавлении пространства имен в [XmlNamespaceManager]. Я просто добавляю этот комментарий на тот случай, если кто-нибудь захочет увидеть небольшой пример кода с документом, имеющим несколько атрибутов [xmlns], с суффиксом и без него. Смотрите здесь: stackoverflow.com/a/38272604/5838538
Теги:
xpath
linq-to-xml

3 ответа

136
Лучший ответ

Если у вас XDocument, проще использовать LINQ-to-XML:

var document = XDocument.Load(fileName);
var name = document.Descendants(XName.Get("Name", @"http://demo.com/2011/demo-schema")).First().Value;

Если вы уверены, что XPath - единственное решение, которое вам нужно:

using System.Xml.XPath;

var document = XDocument.Load(fileName);
var namespaceManager = new XmlNamespaceManager(new NameTable());
namespaceManager.AddNamespace("empty", "http://demo.com/2011/demo-schema");
var name = document.XPathSelectElement("/empty:Report/empty:ReportInfo/empty:Name", namespaceManager).Value;
  • 12
    Я бы сказал, что в большинстве случаев трудно сказать, что linq проще, чем xpath. Например, в этом случае эквивалент LINQ на самом деле не эквивалентен, так как он также получит узлы «Name» под другими узлами (которых сейчас нет, но они могут быть добавлены последующими изменениями в формате файла). Однако ваше решение, безусловно, является правильным.
  • 10
    ПРИМЕЧАНИЕ: использование System.Xml.XPath; довольно важно, потому что XPathSelectElement является методом расширения. Не делай, как я, и игнорируй эту часть;)
Показать ещё 2 комментария
6

XPath 1.0, который реализует MS, не имеет понятия пространства имен по умолчанию. Поэтому попробуйте следующее:

XDocument xdoc = XDocument.Load(@"C:\SampleXML.xml");
XmlNamespaceManager xnm = new XmlNamespaceManager(new NameTable()); 
xnm.AddNamespace("x", "http://demo.com/2011/demo-schema");
Console.WriteLine(xdoc.XPathSelectElement("/x:Report/x:ReportInfo/x:Name", xnm) == null);
  • 8
    Ваш ответ подразумевает, что XPath 2.0, в отличие от XPath 1.0, "*" имеет "идею" пространства имен по умолчанию. Мне не известно о такой новой функции XPath (здесь речь идет о XPath, а не о XSLT или XQuery). Поэтому вы могли бы Пожалуйста, прямо укажите в своем ответе, что вы намекаете?
  • 2
    Я думаю, что он имеет в виду, что если у вас есть документ, который определяет пространство имен, ваш xpath должен включать квалифицированные элементы, т.е. вы не можете делать xnm.AddNamespace (string.Empty, " demo.com/2011/demo-schema" ); а затем xdoc.XPathSelectElement ("/ Report / ReportInfo / Name", xnm) - результат всегда получается нулевым
0

Вы можете использовать пример от Microsoft - для вас без пространства имен:

var e = xdoc.XPathSelectElement("./Report/ReportInfo/Name");     

должен сделать это

Ещё вопросы

Сообщество Overcoder
Наверх
Меню