2012-06-11 8 views
9
XDocument xDocument = XDocument.Load("..."); 
IEnumerable<XElement> elements = xDocument 
    .Element("lfm") 
    .Element("events") 
    .Elements("event"); 

try 
{    
    foreach (XElement elm in elements) 
    { 
     comm.Parameters.AddWithValue("extID", elm.Element("id").Value ?? ""); 
     comm.Parameters.AddWithValue("Title", elm.Element("title").Value ?? ""); 
     comm.Parameters.AddWithValue("HeadlineArtist", 
     elm.Element("artists").Element("headliner").Value ?? ""); 

ma voglio il valore dell'elemento "immagine" con l'attributo "size = large", ho cercato tutta la notte, e questo è il più vicino sono venuto:Trova un XElement con un determinato nome e valore di attributo con LINQ

comm.Parameters.AddWithValue("LargeImage", 
    elm.Descendants("image") 
     .FirstOrDefault(i => (string)i.Attribute("size") == "large").Value); 

campione della parte di risposta XML:

<lfm status="ok"> 
    <events xmlns:geo="http://www.w3.org/2003/01/geo/wgs84_pos#" 
      location="Chicago, United States" page="1" perPage="1" 
      totalPages="341" total="341" festivalsonly="0" tag=""> 
     <event xmlns:geo="http://www.w3.org/2003/01/geo/wgs84_pos#"> 
      <id>3264699</id> 
      <title>Iron And Wine</title> 
      <artists> 
       <artist>Iron And Wine</artist> 
       <artist>Dr. John</artist> 
       <headliner>Iron And Wine</headliner> 
      </artists> 
      <venue> 
       <id>8915382</id> 
       <name>Ravinia Festival</name> 
       <location> 
        <city>Highland Park</city> 
        <country>United States</country> 
        <street>200 Ravinia Park Rd</street> 
        <postalcode>60035</postalcode> 
        <geo:point> 
         <geo:lat>42.15831</geo:lat> 
         <geo:long>-87.778409</geo:long> 
        </geo:point> 
       </location> 
       <url>http://www.last.fm/venue/8915382+Ravinia+Festival</url> 
       <website>http://www.ravinia.org/</website> 
       <phonenumber>847.266.5100</phonenumber> 
       <image size="small">http://userserve-ak.last.fm/serve/34/63026487.jpg</image> 
       <image size="medium">http://userserve-ak.last.fm/serve/64/63026487.jpg</image> 
       <image size="large">http://userserve-ak.last.fm/serve/126/63026487.jpg</image> 
       <image size="extralarge">http://userserve-ak.last.fm/serve/252/63026487.jpg</image> 
+0

Allora, qual è il problema? Ciò sembra soddisfacente, ma dipende da cosa 'elm' è (non ti mostri come ho ottenuto da xDocument in elm). – HackedByChinese

+0

Otterrai 'NullReferenceException' quando non viene trovato alcun elemento con quell'attributo. – MarcinJuraszek

risposta

25

Prova

XElement result = elm.Descendants("image") 
    .FirstOrDefault(el => el.Attribute("size") != null && 
         el.Attribute("size").Value == "large"); 
if (result != null) { 
    process result.Value ... 
} 

A partire con C# 6.0 (VS 2015), si può scrivere:

XElement result = elm.Descendants("image") 
    .FirstOrDefault(el => el.Attribute("size")?.Value == "large"); 
if (result != null) { 
    process result.Value ... 
} 

Un non-ovvia alternativa (come @RandRandom sottolineato) è quello di gettare l'attributo string:

XElement result = elm.Descendants("image") 
    .FirstOrDefault(el => (string)el.Attribute("size") == "large"); 
if (result != null) { 
    process result.Value ... 
} 

Funziona, perché a causa di XAttribute Explicit Conversion (XAttribute to String).

+0

per somereason questo non funzionava all'inizio, funziona benissimo anche se ora! –

+1

Invece di? .Valore o .Valore per iniziare, puoi scrivere in questo modo: (stringa) el.Attribute ("size") == "large" (funziona in tutte le versioni) –

11

è possibile utilizzare XPathSelectElement extension method

var node = elm.XPathSelectElement("descendant::image[@size='large']"); 
if (node!=null) 
{ 
    var path = node.Value; 
} 
+1

+1. Soluzione semplice –

+1

Sorprendente risposta !. Testato e funziona anche per .NET Core/.NET Standard 1.6! –