Linq ถึง Xml: การเลือกองค์ประกอบหากค่าแอตทริบิวต์เท่ากับค่าโหนดใน IEnumerable‹XElement›

ฉันสร้างวัตถุ IEnumerable ที่มีเพียงโหนดที่ฉันต้องการจากไฟล์ xml:

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

จึงเป็นการรวบรวมสิ่งเหล่านี้:

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

ฉันต้องการคว้าเฉพาะผู้ใช้ที่แอตทริบิวต์ userid ก็เป็นโหนด userid ภายในคอลเลกชัน rosters ด้วย

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

แต่มันทำให้ฉันมีข้อผิดพลาด:

อาร์กิวเมนต์ประเภทสำหรับเมธอด 'System.Linq.Enumerable.Contains(System.Collections.Generic.IEnumerable, TSource)' ไม่สามารถอนุมานได้จากการใช้งาน ลองระบุอาร์กิวเมนต์ประเภทอย่างชัดเจน

แนวทางของฉันมีอะไรผิดปกติ?


person lolcat    schedule 22.02.2009    source แหล่งที่มา


คำตอบ (1)


ปัญหาหนึ่งที่ฉันเห็นคือในข้อมูลโค้ดล่าสุด ...Elements("userid") ส่งคืนรายการวัตถุ XElement ที่ไม่สามารถมีสตริงที่ส่งคืนโดยคุณสมบัติ Value สิ่งนี้ควรจะได้ผล...

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

อย่างไรก็ตาม ฉันจะทำสิ่งนี้โดยใช้แบบสอบถามเข้าร่วม .. Select user join rosters on userid
มันจะเป็นแบบนี้

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

เอาท์พุต:

User on roster       on Roster R1
person Gishu    schedule 22.02.2009