当父节点等于LINQ to XML中的特定值时,如何选择子节点

时间:2012-05-01 13:46:47

标签: c# asp.net xml linq robots.txt

我正在构建的应用程序的一部分需要从XML文档生成robots.txt文件。

我有这样的XML:

<root>
  <Robots>
    <UserAgents>
      <UserAgent>*</UserAgent>
      <Disallow>
        <Item>/wibble/</Item>
        <Item>/wobble/</Item>
      </Disallow>
    </UserAgents>
    <UserAgents>
      <UserAgent>Google</UserAgent>
      <Disallow>
        <Item>/</Item>
      </Disallow>
    </UserAgents>
  </Robots>
</root>

如何选择UserAgent =“*”的“项目”?

换句话说,我只想在父节点等于某个值时才选择子节点。父节点中的值将是唯一的。

我在ASP.net中使用C#。

2 个答案:

答案 0 :(得分:4)

听起来像你想要的东西:

var query = from agent in doc.Descendants("UserAgent")
            where (string) agent == "*"
            from item in agent.Parent.Elements("Disallow").Elements("Item") 
            select item;

请注意,第二个from仅用于展平结果,因此它不是一系列序列。

可替换地:

var query = doc.Descendants("UserAgent")
               .Where(agent => (string) agent == "*")
               .SelectMany(agent => agent.Parent.Elements("Disallow")
                                                .Elements("Item"));

或者通过将选定的“顶部”节点更改为UserAgents,您可以避免使用父步骤:

var query = doc.Descendants("UserAgents")
               .Where(agents => (string) agents.Element("UserAgent") == "*")
               .SelectMany(agent => agent.Elements("Disallow").Elements("Item"));

请注意,如果您在单个UserAgent元素下有多个UserAgents元素,则不会产生相同的结果。

(你能改变XML架构,顺便说一下?看起来你应该能够简化这个......)

答案 1 :(得分:1)

var query = from x in doc.Root.Element("Robots").Elements("UserAgents")
            where (string)x.Element("UserAgent") == "*"
            from y in x.Element("Disallow").Elements("Item")
            select y;