使用linq的XML多对多连接

本文关键字:连接 XML linq 使用 | 更新日期: 2023-09-27 18:06:14

我可能在编写适当的代码或提出问题时出错,但我正在寻求以下帮助。

我有一个多对多关系的xml文件,如下所示:

<ShowRoom>
  <Cars>
    <Car id="1" number="001" name="MyCar" >
      <Parts>
        <Part id="1" />
        <Part id="2" />
      </Parts>
    </Car>
    <Car id="2" number="002" name="YourCar" >
      <Parts>
        <Part id="2" />
        <Part id="3" />
        <Part id="4" />
      </Parts>
    </Car>
  </Cars>
  <Parts>
    <Part id="1" name="Tyre" active="true"/>
    <Part id="2" name="Window" active="true"/>
  </Parts>
</ShowRoom>

与该对象对应的是follow

public class Car
{
    private int _id;
    public int Id
    {
        get { return _id; }
        set { _id = value; }
    }
    private string _number;
    public string Number
    {
        get { return _number; }
        set { _number = value; }
    }
    private string _name;
    public string Name
    {
        get { return _name; }
        set { _name = value; }
    }
    private List<Parts> _listParts;
    public List<Parts> ListParts
    {
        get { return _listParts; }
        set { _listParts = value; }
    }
}

我正在尝试的是读取xml和汽车和零件之间的连接,所以我将获得汽车对象及其所有零件的详细信息

  IEnumerable<XElement> xCar = xdoc.Element("ShowRoom").Element("Cars").Elements("Car");
                IEnumerable<XElement> xPart = xdoc.Element("ShowRoom").Element("Parts").Elements("Part");
                IEnumerable<Car> car = xCar.Join(
                    xPart,
                    c => c.Element("Parts").Element("Part").Attribute("id").Value,
                    p => p.Attribute("id").Value,
                    (c, p) =>
                    new Cars
                    {
                        Id = int.Parse(c.Attribute("id").Value),
                        Number = (string)c.Attribute("number").Value,
                        Name = (string)c.Attribute("name").Value,
                        ListParts =  
                        { 
                                new Parts
                                    {
                                        Id = int.Parse(p.Attribute("id").Value),
                                        Name = (string)p.Attribute("name").Value,
                                        Active = bool.Parse(p.Attribute("active").Value)
                                    }
                        }
                    });

这里的问题是,我不知道如何得到所有部件对汽车像上面的代码,这显然是错误的

                        ListParts =  
                        { 
                                new Parts
                                    {
                                        Id = int.Parse(p.Attribute("id").Value),
                                        Name = (string)p.Attribute("name").Value,
                                        Active = bool.Parse(p.Attribute("active").Value)
                                    }

那么如何使用相同的代码来完成它,或者如果有其他好的方法,请与我分享。

谢谢。

使用linq的XML多对多连接

我强烈建议建立一个零件字典,然后建立汽车列表:

Dictionary<int, Part> partsById = doc.Root
    .Element("Parts")
    .Elements("Part")
    .Select(p => new Parts
            {
                Id = (int) p.Attribute("id"),
                Name = (string) p.Attribute("name"),
                Active = (bool) p.Attribute("active")
            })
    .ToDictionary(part => part.Id);

请注意,我已将Parts重命名为Part,因为每个对象仅代表单个部分-并且我已删除了所有解析,使用XAttribute上的显式转换。

当你创建你的汽车列表时,你可以使用:

...
ListParts = c.Element("Parts")
             .Elements("Part")
             .Select(p => partsById[(int) p.Attribute("Id")])
             .ToList();

您还应该研究自动实现的属性,这将把您的Car类减少到只有几行代码:

public class Car
{
    public int Id { get; set; }
    public string Number { get; set; }
    public string Name { get; set; }
    public List<Parts> ListParts { get; set; }
}