LINQ to XML: selezione di elementi se un valore di attributo è uguale a un valore di nodo in un IEnumerable <XElement>


8

creo un oggetto IEnumerable che contiene solo i nodi che voglio da un file XML:

IEnumerable<XElement> rosters = XDocument.Load("roster.xml") 
             .Elements("rosterlist") 
             .Elements("roster") 
             .Where(w => w.Element("division") 
                 .Value 
                 .Equals("SUPER AWESOME DIVISION")); 

Quindi è una raccolta di questi:

<rosterlist> 
    <roster> 
     <userid>1</userid> 
     <name></name> 
     <etc></etc> 
    </roster> 
    <roster> 
     <userid>2</userid> 
     <name></name> 
     <etc></etc> 
    </roster> 
</rosterlist> 

voglio afferrare solo gli utenti in cui l'attributo userid è anche un nodo userid all'interno della collezione rosters.

IEnumerable<XElement> users = XDocument.Load("user.xml") 
             .Elements("userlist") 
             .Elements("user") 
             .Where(w => rosters.Elements("userid") 
                  .Contains(w.Attribute("userid").Value)); 

Ma mi sta dando un errore:

Gli argomenti di tipo per il metodo 'System.Linq.Enumerable.Contains (System.Collections.Generic.IEnumerable, TSource)' non può essere dedotta dalla utilizzo. Prova a specificare esplicitamente gli argomenti del tipo.

Cosa c'è di sbagliato nel mio approccio?

6

Un problema che vedo è che nell'ultimo snippet di codice, ...Elements("userid") restituisce un elenco di oggetti XElement che non può contenere una stringa restituita dalla proprietà Value. Questo dovrebbe funzionare ...

IEnumerable<XElement> rosters = obRoot.Elements("rosterlist").Elements("roster"); 
var rosterUserIds = (rosters.Elements("userid").Select(r => r.Value)); 
IEnumerable<XElement> users = obRoot.Elements("userlist").Elements("user") 
       .Where(u => rosterUserIds.Contains(u.Attribute("userid").Value)); 

Comunque avrei fatto questo utilizzando un join query di .. Selezionare l'utente unirsi roster su userid
Sarebbe più o meno così

string sXml = @" 
<root> 
<rosterlist> 
    <roster> 
     <userid>1</userid> 
     <name>R1</name> 
     <etc></etc> 
    </roster> 
    <roster> 
     <userid>2</userid> 
     <name>R2</name> 
     <etc></etc> 
    </roster> 
</rosterlist> 
<userlist> 
    <user userid='1'> 
     <name>User on roster</name> 
    </user> 
    <user userid='5'> 
     <name>User not on roster</name> 
    </user> 
</userlist> 
</root> 

"; 

XElement obRoot = XElement.Parse(sXml); 
var results = from user in obRoot.Elements("userlist").Elements("user") 
    join roster in obRoot.Elements("rosterlist").Elements("roster") 
    on user.Attribute("userid").Value equals roster.Element("userid").Value 
    select new {Name=user.Element("name").Value, RosterName=roster.Element("name").Value} ; 

foreach (var v in results) 
{ 
    Console.WriteLine("{0, -20} on Roster {1, -20}", v.Name, v.RosterName); 
} 

Uscite:

User on roster  on Roster R1