如何使用 XDocument 和 Linq 读取与子注释连接的父属性列表

本文关键字:连接 注释 列表 属性 XDocument 何使用 Linq 读取 | 更新日期: 2023-09-27 18:32:23

我是Linq和XDocument的新手,我在任何地方都找不到类似的解决方案问题,所以这里是:

这是我尝试解析的 XML:

<environment>
 <objectBuilder>
  <service id="1">
   <!--Service:"MagicService">
  </service>
  <service id="2">
   <!--Service:"RubbishService">
  </service>
  <service id="3">
   <!--Service:"MediocreService">
  </service>
 </objectBuilder>
</environment>

我想使用一个如下所示的列表:

("1 MagicService","2 RubbishService","3 MediocreService")

我已经尝试将注释和 id 读取到两个不同的列表中,然后将它们组合在一起,但这不是一个足够好的解决方案,因为在我的现实世界中,可能有没有注释名称的服务。

我当前的解决方案如下所示:

List<string> comments = doc.Elements("Environment")
                        .Elements("objectBuilder")
                        .Elements("service")
                        .Nodes()
                        .OfType<XComment>()
                        .Select(c => c.Value)
                        .Where(c=> c.StartsWith("Service"))
                        .ToList());
List<string> serviceGuides = doc.Elements("Environment")
                        .Elements("objectBuilder")
                        .Elements("service")
                        .Attributes(XName.Get("id"))
                        .Select(c => c.Value)
                        .ToList());

然后,如果列表大小相同,我将通过一个for循环并逐字地连接它们。 一定有更优雅的解决方案吗?

感谢您的帮助,如果这是一个重复/愚蠢的问题,对不起

如何使用 XDocument 和 Linq 读取与子注释连接的父属性列表

这是一种可能的方法:

var result = (from service in doc.Elements("environment")
                                 .Elements("objectBuilder")
                                 .Elements("service")
                     //get service id attribute
              select (string)service.Attribute("id")
                      + " " +
                     //get the first comment node which value starts with "Service"
                     //and then extract the value between first quotes pair
                      service.Nodes()
                             .OfType<XComment>()
                             .Select(c => c.Value)
                             .FirstOrDefault(c => c.StartsWith("Service"))
                             .Split('"')[1]
             ).ToList();
foreach (var r in result)
{
    Console.WriteLine(r);
}

dotnetfiddle demo

输出:

1 MagicService
2 RubbishService
3 MediocreService

这应该有效。

var serviceElements = xDoc.Elements("environment")
    .Elements("objectBuilder")
    .Elements("service")                
    .ToList();
var result = (from element in serviceElements
    let attribute = element.Attribute("id").Value
    let name = element
        .Nodes()
        .OfType<XComment>()
        .Select(c => c.Value)
        .FirstOrDefault(c => c.StartsWith("Service"))
        .Split('"')[1]
    select string.Format("{0} {1}", attribute, name)).ToList();

Resharper非常适合这样的事情。您可以编写循环,resharper 将帮助您将其转换为 linq。您可以轻松地组合上面的两个表达式。您可能还希望添加空引用检查。