2013-02-19 104 views
1

我有以下XML:在xml中查找具有特定属性的元素?

<?xml version="1.0" encoding="utf-8" ?> 
<layout> 
    <menu name="Employees" url="Employees.aspx" admin="0"> 
    </menu> 
    <menu name="Projects" url="Projects.aspx" admin="1"> 
    </menu> 
    <menu name="Cases" url="Cases.aspx" admin="1"> 
    </menu> 
    <menu name="CaseView" url="CaseView.aspx" admin="1" hidden="1" parent="Projects"> 
    </menu> 
    <menu name="Management" url="" admin="1"> 
    <item name="Groups" url="Groups.aspx" admin="1" parent="Management"/> 
    <item name="Statuses" url="Statuses.aspx" admin="1"/> 
    </menu> 
</layout> 

这里我有CaseView和组都有一个“父”属性。

目前我重复这样的:

IEnumerable<XElement> menus = 
      doc.Element("layout").Elements(); 

      foreach (var menu in menus) 
      { 
        string name = menu.Attribute("name").Value; 
       string active = ""; 
       string url = menu.Attribute("url").Value; 
       if(activePage == url) 
       { 
        active = "class=\"active\""; 
       } 
       ... 

我要的是:

 if(activePage == url || ActiveIsChildOf(name, activePage)) 
     { 
      active = "class=\"active\""; 
     } 

本质上这种方法需要找到如果与activePage元素作为其url属性存在。如果有,请查看它是否具有父属性;如果是这样,请检查父==的名称。

有没有某种方法来查找元素的属性或东西? 例如:

XElement e = doc.GetByAttribute("url",activePage) 

感谢

回答

1

由于您使用Linq to XML,因此您可以使用Descendants方法 - 它将返回所有子元素,而不仅仅是直接子元素。之后,您可以使用LINQ来过滤结果。

XDocument doc; 
string activePage; 
var activeMenu = doc.Descendants("menu") 
    .FirstOrDefault(o => o.Attribute("url").Value == activePage); 

您可能需要检查是否o.Attribute("url")没有返回null(当属性不存在它),如果你不能保证源XML没有为所有的菜单等元素属性。

您也可以跳过参数到Descendants()来检查所有元素 - 在您的示例数据中可以检查菜单和项目元素。例如:

var activeMenu = doc.Descendants() 
    .Where(o => o.Name == "menu" || o.Name == "item") 
    .FirstOrDefault(o => o.Attribute("url").Value == activePage); 
0

您可以简单地使用xPath。这是一个查询语言XML

您可以制定这样的事情:

var xDoc = new XmlDocument(); 
xDoc.Load("XmlFile.xml"); 

//Fetch your node here 
XmlNode = xDoc.SelectSingleNode(/layout/menu[@url='activepage'][1]); 

它返回一组节点和索引1是获取给定的第一个节点。

如果您想要所有匹配的节点,则始终可以使用xDoc.SelectNodes


由于您使用LINQ你可以简单地包括System.Xml.XPathXPathSelectElementXPathSelectElements选择的节点。

1

如果XPath是太神秘,你可以使用LINQ:

IEnumerable<XElement> hits = 
     (from el in XMLDoc.root.Elements("item") 
     where (string)el.Attribute("url") == activePage 
     select el); 

或像这样:

XElement xml = XElement.Load(file); 
XElement xele = xml.Elements("item").FirstOrDefault(e => ((string)e.Attribute("url")) == activePage); 
if(null != xele) 
{ 
    // do something with it 
} 

而且你可能希望它不区分大小写:

XElement xml = XElement.Load(file); 
XElement xele = xml.Elements("item").FirstOrDefault(e => StringComparer.OrdinalIgnoreCase.Equals((string)e.Attribute("url") , activePage)); 
if(null != xele) 
{ 
    // do something with it 
} 

如果你想要菜单和项目,使用这个:

XElement xml = XElement.Load(file); 
XElement xele = xml.Elements().FirstOrDefault(e => StringComparer.OrdinalIgnoreCase.Equals((string)e.Attribute("url") , activePage)); 
if(null != xele) 
{ 
    // do something with it 
} 
+0

Niice不知道这一点。对我来说,LINQ和Lambdas的waaaaaay比XPath更神秘,但是因为他在上面的例子中使用了它,所以我猜这对他来说可能是一个更简单的解决方案。 – phadaphunk 2013-02-19 17:15:25

+0

这将返回所有或只'元素'元素我也有'菜单'的元素。 – jmasterx 2013-02-19 17:20:07

+0

第二个最后一个只找到属性url =“activePage”类型的“item”类型的第一个元素,第一个元素的类型为“item”,属性url =“activePage”,最后一个是任何类型的第一个元素具有属性activepage – 2013-02-19 17:38:25

0

你可以做到这一点的XPath:

doc.SelectNodes("//*[@url='" + activePage + "']") 

它会返回activePageurl属性都文件的项目。

0

不区分大小写的搜索示例,将XML转换为词典:

Dim expandos = XDocument.Parse(Request("Xml")).Root.Elements.Select(
    Function(e) 
     Dim expando As Object = New ExpandoObject, 
      dic = e.Attributes.ToDictionary(Function(a) a.Name.LocalName, Function(a) a.Value, 
       StringComparer.InvariantCultureIgnoreCase) 
     expando.PedidoId = dic("PedidoId") 
     expando.FichaTecnicaModeloId = dic("FichaTecnicaModeloId") 
     expando.Comodo = dic("Comodo") 
     expando.Cliente = dic("Cliente") 
     Return expando 
    End Function) 
相关问题