2010-04-07 16 views
6

Sto cercando di utilizzare XPath per selezionare gli elementi che hanno un aspetto con valori Location, ma attualmente i miei tentativi anche solo per selezionare tutti gli elementi non riescono: Il sistema riporta felicemente che ha trovato 0 elementi, quindi restituisce (invece i nodi dovrebbero essere elaborati da un ciclo foreach). Apprezzerei molto l'aiuto sia a fare la mia query originale o semplicemente a far funzionare XPath a tutti.C# XPath Not Finding Anything

XML

<?xml version="1.0" encoding="UTF-8" ?> 
<Collection Name="My Collection" SchemaVersion="1.0" xmlns="http://schemas.microsoft.com/collection/metadata/2009" xmlns:p="http://schemas.microsoft.com/livelabs/pivot/collection/2009" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema"> 
<FacetCategories> 
    <FacetCategory Name="Current Address" Type="Location"/> 
    <FacetCategory Name="Previous Addresses" Type="Location" /> 
</FacetCategories> 
    <Items> 
     <Item Id="1" Name="John Doe"> 
      <Facets> 
       <Facet Name="Current Address"> 
        <Location Value="101 America Rd, A Dorm Rm 000, Chapel Hill, NC 27514" /> 
       </Facet> 
       <Facet Name="Previous Addresses"> 
        <Location Value="123 Anywhere Ln, Darien, CT 06820" /> 
        <Location Value="000 Foobar Rd, Cary, NC 27519" /> 
       </Facet> 
      </Facets> 
     </Item> 
    </Items> 
</Collection> 

C#

public void countItems(string fileName) 
{ 
    XmlDocument document = new XmlDocument(); 
    document.Load(fileName); 
    XmlNode root = document.DocumentElement; 
    XmlNodeList xnl = root.SelectNodes("//Item"); 
    Console.WriteLine(String.Format("Found {0} items" , xnl.Count)); 
} 

C'è di più al metodo di questo, ma dal momento che questo è tutto ciò che viene eseguito Sto assumendo il problema si trova qui. Chiamando root.ChildNodes restituisce esattamente FacetCategories e Items, quindi sono completamente in perdita.

Grazie per il vostro aiuto!

risposta

17

L'elemento radice ha uno spazio dei nomi. Avrai bisogno di aggiungere un risolutore di namespace e prefisso gli elementi nella tua query.

This article spiega la soluzione. Ho modificato il tuo codice in modo che ottenga 1 risultato.

public void countItems(string fileName) 
{ 
    XmlDocument document = new XmlDocument(); 
    document.Load(fileName); 
    XmlNode root = document.DocumentElement; 

    // create ns manager 
    XmlNamespaceManager xmlnsManager = new XmlNamespaceManager(document.NameTable); 
    xmlnsManager.AddNamespace("def", "http://schemas.microsoft.com/collection/metadata/2009"); 

    // use ns manager 
    XmlNodeList xnl = root.SelectNodes("//def:Item", xmlnsManager); 
    Response.Write(String.Format("Found {0} items" , xnl.Count)); 
} 
6

Perché hai un namespace XML sul nodo principale, non v'è alcuna cosa come "voce" nel documento XML, solo "[namespace]: Oggetto", in modo che quando la ricerca di un nodo con XPath, è necessario specificare lo spazio dei nomi.

Se non ti piace, puoi utilizzare la funzione di nome locale() per trovare tutti gli elementi il ​​cui nome locale (la parte del nome diversa dal prefisso) è il valore che stai cercando. È un po 'brutta sintassi, ma funziona.

XmlNodeList xnl = root.SelectNodes("//*[local-name()='Item']");