在xml中查找具有特定属性的元素?
问题描述:
我有以下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)
感谢
答
由于您使用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);
答
您可以简单地使用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.XPath
与XPathSelectElement
或XPathSelectElements
选择的节点。
答
如果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
}
答
你可以做到这一点的XPath:
doc.SelectNodes("//*[@url='" + activePage + "']")
它会返回有activePage
为url
属性都文件的项目。
答
不区分大小写的搜索示例,将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)
Niice不知道这一点。对我来说,LINQ和Lambdas的waaaaaay比XPath更神秘,但是因为他在上面的例子中使用了它,所以我猜这对他来说可能是一个更简单的解决方案。 – phadaphunk 2013-02-19 17:15:25
这将返回所有或只'元素'元素我也有'菜单'的元素。 – jmasterx 2013-02-19 17:20:07
第二个最后一个只找到属性url =“activePage”类型的“item”类型的第一个元素,第一个元素的类型为“item”,属性url =“activePage”,最后一个是任何类型的第一个元素具有属性activepage – 2013-02-19 17:38:25