2010-03-22 1 views
5

Sto riscontrando un problema nel far funzionare una query LINQ. Ho questo XML:Interrogazione nidificata da Linq a XML

<devices> 
    <device id ="2142" name="data-switch-01"> 
    <interface id ="2148" description ="Po1"/> 
    </device> 
    <device id ="2302" name="data-switch-02"> 
    <interface id ="2354" description ="Po1"/> 
    <interface id ="2348" description ="Gi0/44" /> 
    </device> 
</devices> 

E questo codice:

var devices = from device in myXML.Descendants("device") 
       select new 
       { 
        ID = device.Attribute("id").Value, 
        Name = device.Attribute("name").Value, 
       }; 

foreach (var device in devices) 
{ 
    Device d = new Device(Convert.ToInt32(device.ID), device.Name); 

    var vIfs = from vIf in myXML.Descendants("device") 
        where Convert.ToInt32(vIf.Attribute("id").Value) == d.Id 
        select new 
        { 
         ID = vIf.Element("interface").Attribute("id").Value, 
         Description = vIf.Element("interface").Attribute("description").Value, 
        }; 
    foreach (var vIf in vIfs) 
    { 
     DeviceInterface di = new DeviceInterface(Convert.ToInt32(vIf.ID), vIf.Description); 
     d.Interfaces.Add(di); 
    } 

    lsDevices.Add(d); 
} 

mio oggetto dispositivo contiene un elenco di DeviceInterfaces di cui ho bisogno per popolare dal XML. Al momento il mio codice popola solo la prima interfaccia, quelli successivi vengono ignorati e non riesco a capire perché.

Apprezzerei anche eventuali commenti sul fatto che questo sia il modo giusto per farlo. I cicli foreach nidificati sembrano un po 'confuso per me

Acclamazioni

risposta

12
IEnumerable<Device> devices = 
    from device in myXML.Descendants("device") 
    select new Device(device.Attribute("id").Value, device.Attribute("name").Value) 
    { 
    Interfaces = (from interface in device.Elements("Interface") 
        select new DeviceInterface(
         interface.Attribute("id").Value, 
         interface.Attribute("description").Value) 
       ).ToList() //or Array as you prefer 
    } 

Il punto fondamentale è che si fa una sorta di "sub-SELECT" sul dispositivo (che è un Descendant), alla ricerca di tutti gli Interface elementi che contiene.

Crea un nuovo DeviceInterface per ogni "interfaccia" sotto ciascun dispositivo.

+0

Grazie che sembra molto più bello, ho deciso di dargli un colpo in seguito :) – user299342

+0

Yep che è esatte, evviva! – user299342

1

rapido e sporco

var query = from device in document.Descendants("device") 
      select new 
      { 
       ID = device.Attribute("id").Value, 
       Name = device.Attribute("name").Value, 
       Interfaces = from deviceInterface in device.Descendants("interface") 
          select new 
          { 
           ID = deviceInterface.Attribute("id").Value, 
           Description = deviceInterface.Attribute("description") 
          } 
      }; 
+0

È ancora necessario iterare su 'query' per creare (o popolare) un' Elenco '(vedere' lsDevices' nel codice postato). –