como usar XPath com XDocument?

109

Há uma questão semelhante, mas parece que a solução não funcionou no meu caso: Estranheza com XDocument, XPath e namespaces

Aqui está o XML com o qual estou trabalhando:

<?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>

E abaixo está o código que pensei que deveria estar funcionando, mas não funcionou ...

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);

Alguém tem alguma idéia? Obrigado.

Jojo
fonte
1
Veja a outra resposta abaixo, ela não funciona porque a implementação XPath 1.0 não consegue lidar com um prefixo vazio
Paul Hatcher
1
Como já foi dito aqui, não use um prefixo vazio ao adicionar um Namespace ao [XmlNamespaceManager]. Estou apenas adicionando este comentário caso alguém queira ver um pequeno exemplo de código com um documento que tem vários atributos [xmlns], com e sem um sufixo. Veja aqui: stackoverflow.com/a/38272604/5838538
Jelgab

Respostas:

158

Se você tiver o XDocument, é mais fácil usar o LINQ-to-XML:

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

Se você tiver certeza de que XPath é a única solução de que você precisa:

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;
Alex Aza
fonte
13
Eu diria que é difícil dizer que linq é mais fácil do que xpath na maioria dos casos. Por exemplo, neste caso, o equivalente LINQ não é realmente equivalente, pois também obteria nós "Nome" em outros nós (que não existem agora, mas podem ser adicionados por alterações posteriores no formato do arquivo). No entanto, a sua solução é certamente a certa.
Marco Mp
12
NOTA: usando System.Xml.XPath; é muito importante porque o XPathSelectElement é um método de extensão. Não faça o que eu fiz e ignore essa parte;)
Mark van Straten
7
XPath ainda é útil porque permite contextualizar seus relacionamentos entre pais e filhos. Por exemplo, se você quisesse obter / Banana / Banana / Banana em vez de obter todas as Bananas
Sebastian Patten
2
"vazio" é um pouco enganador e confuso aqui. Você pode usar qualquer coisa, exceto, com XPath, String.Empty (como o autor da pergunta descobriu). "demo" seria mais apropriado para o exemplo.
Tom Blodget
7

XPath 1.0, que é o que a MS implementa, não tem a ideia de um namespace padrão. Então tente isso:

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);
Richard Schneider
fonte
8
Sua resposta implica que XPath 2.0, em contraste com XPath 1.0 "* tem" uma idéia "de um namespace padrão. Não estou ciente desse novo recurso XPath (estamos falando de XPath aqui, não de XSLT ou XQuery). Portanto, você poderia , por favor, mencione explicitamente em sua resposta o que você está
querendo dizer
2
Acho que ele está querendo dizer que se você tem um documento que define um namespace, seu xpath deve incluir elementos qualificados, ou seja, você não pode fazer xnm.AddNamespace (string.Empty, " demo.com/2011/demo-schema" ); e em seguida xdoc.XPathSelectElement ("/ Report / ReportInfo / Name", xnm) - o resultado sempre sai nulo
Paul Hatcher
3

você pode usar o exemplo da Microsoft - para você sem namespace:

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

deveria fazer isso

Bernhard
fonte
não funciona para mim
user1623521