Linq zu Xml: Auswählen von Elementen, wenn ein Attributwert einen Knotenwert in einer IEnumerable <XElement>


8

gleich ich ein IEnumerable-Objekt erstellen, die nur die Knoten enthalten ich aus einer XML-Datei will:

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

es ist eine Sammlung also von diese:

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

ich nur die Benutzer greifen will, wo das userid Attribut ist auch ein userid Knoten innerhalb der rosters Sammlung.

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

Aber es gibt mir eine Fehlermeldung:

Die Art Argumente für die Methode 'System.Linq.Enumerable.Contains (System.Collections.Generic.IEnumerable, TSource)' kann nicht aus der abgeleitet werden Verwendung. Versuchen Sie, die Typargumente explizit anzugeben.

Was mit meinem Ansatz ist falsch?

6

Ein Problem, das ich sehe, ist, dass im letzten Code-Snippet ...Elements("userid") eine Liste von XElement-Objekten zurückgibt, die keinen von der Value-Eigenschaft zurückgegebenen String enthalten können. Dies sollte funktionieren ...

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

aber ich dies tun würde durch eine Join-Abfrage mit .. Wählen Sie Benutzer beitreten Dienstpläne auf Benutzer-ID
es so etwas wie diese

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

Ausgänge gehen würde:

User on roster  on Roster R1