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

109

Есть аналогичный вопрос, но, похоже, в моем случае решение не сработало: странность с 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);

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

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

Ответы:

158

Если у вас есть 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;
Алекс Аза
источник
13
Я бы сказал, что в большинстве случаев трудно сказать, что linq проще, чем xpath. Например, в этом случае эквивалент LINQ на самом деле не эквивалентен, поскольку он также получит узлы «Имя» под другими узлами (которых сейчас нет, но они могут быть добавлены более поздними изменениями в формате файла). Однако ваше решение, безусловно, правильное.
Marco Mp
12
ПРИМЕЧАНИЕ: using System.Xml.XPath; очень важен, потому что XPathSelectElement является методом расширения. Не делай того, что я сделал, и игнорируй эту часть;)
Марк ван Стратен
7
XPath по-прежнему полезен тем, что позволяет контекстуализировать отношения между родителями и детьми. Например, если вы хотите добраться до / Banana / Banana / Banana вместо того, чтобы получать каждый банан
Себастьян Паттен
2
"пустой" здесь немного вводит в заблуждение и сбивает с толку. Вы можете использовать что угодно, кроме XPath, String.Empty (как обнаружил спрашивающий). "демо" было бы более подходящим для примера.
Том Блоджет
7

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);
Ричард Шнайдер
источник
9
Ваш ответ подразумевает, что XPath 2.0, в отличие от XPath 1.0 "*, имеет" представление "о пространстве имен по умолчанию. Мне не известно о такой новой функции XPath (здесь мы говорим XPath, а не XSLT или XQuery). Поэтому не могли бы вы пожалуйста, четко укажите в своем ответе, что вы имеете в виду?
Димитр Новачев 02
2
Я думаю, он имеет в виду, что если у вас есть документ, который определяет пространство имен, ваш xpath должен включать квалифицированные элементы, то есть вы не можете делать xnm.AddNamespace (string.Empty, " demo.com/2011/demo-schema" ); а затем xdoc.XPathSelectElement ("/ Report / ReportInfo / Name", xnm) - результат всегда будет нулевым
Пол Хэтчер
4

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

using System.Xml.Linq;
using System.Xml.XPath;
var e = xdoc.XPathSelectElement("./Report/ReportInfo/Name");     

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

Бернхард
источник
у меня не работает
user1623521