C#使用LINQ to XML读取具有相同名称的多个元素

时间:2011-06-27 06:14:01

标签: c# .net xml linq xpath

有没有更好的方法呢? 我必须得到两个属性值。 XML总是只有这两个属性。

我的xml:

<Template name="filename.txt">
  <Property name="recordSeparator">\r\n</Property>
  <Property name="fieldCount">16</Property>
</Template>

的LINQ:

            var property = from template in xml.Descendants("Template")
                       select new
                                  {
                                      recordDelim = template.Elements("Property").Where(prop => prop.Attribute("name").Value == "recordSeparator")
                                                    .Select(f => new { f.Value }),
                                      fieldCount = template.Elements("Property").Where(prop => prop.Attribute("name").Value == "fieldCount")
                                                    .Select(f => new { f.Value })
                                  };

2 个答案:

答案 0 :(得分:3)

“更好的方式”取决于你想要达到的目标 - 性能,简单性等?

我想我会创建一个包含您尝试使用匿名类的内容的类。

public class Item {
    public String Separator { get; set; }
    public int FieldCount { get; set; }
}

然后我会将LINQ修改为:

var templates = from template in xml.Descendants("Template")
                let children = template.Elements("Property")
                select new Item() {
                    Separator = children.First(tag=>tag.Attribute("name").Value == "recordSeparator").Value,
                    FieldCount = Int32.Parse(children.First(tag=>tag.Attribute("name").Value == "fieldCount").Value)
                };

List<Item> items = templates.ToList();

请注意,如果您的Template标记不包含两个属性标记,每个属性都具有指定的属性,则会导致NullReference异常。

如果它不是数字,它也会在解析FieldCount中的整数时抛出异常。

<强>观

如果生成的xml是您自己的,并且您可以更改它的格式,为什么不执行以下操作:

<Template>
  <Name>filename.txt</Name>
  <RecordSeparator>\r\n</RecordSeparator>
  <FieldCount>16</FieldCount>
</Template>

它更容易阅读和解析,而且有点短。

最后,我认为我会这样做:

有这个课程:

public class Item 
{
   public String FileName { get; set; }
   public String Separator { get; set; }
   public int FieldCount { get; set; }
}

和这个私有方法:

private Item GetItemFromTemplate(XElement node) 
{
    return new Item() {
        FileName = node.Element("Name").Value,
        Separator = node.Element("RecordSeparator").Value,
        FieldCount = Int32.Parse(node.Element("FieldCount").Value)
    }; 
}    

我可以在代码中做到:

XDocument doc = XDocument.Load("myfile.txt");

List<Item> items = (from template in doc.Elements("Template")
                   select GetItemFromTemplate(template)).ToList();

答案 1 :(得分:1)

这个效率更高一点:

var properties =
    from template in xml.Descendants("Template")
    let propertyNodes = template.Elements("Property")
        .Select(arg => new { Name = arg.Attribute("name").Value, Value = arg.Value })
    select
        new
        {
            recordDelim = propertyNodes.Single(arg => arg.Name == "recordSeparator").Value,
            fieldCount = propertyNodes.Single(arg => arg.Name == "fieldCount").Value
        };

如果您总是有一个Template节点:

var propertyNodes = xml.XPathSelectElements("/Template/Property")
    .Select(arg => new { Name = arg.Attribute("name").Value, arg.Value })
    .ToList();

var properties =
    new
    {
        recordDelim = propertyNodes.Single(arg => arg.Name == "recordSeparator").Value,
        fieldCount = propertyNodes.Single(arg => arg.Name == "fieldCount").Value
    };