我有一个XML,例如
<?xml version="1.0" encoding="utf-8"?>
<A1>
<B2>
<C3 id="1">
<D7>
<E5 id="abc" />
</D7>
<D4 id="1">
<E5 id="abc" />
</D4>
<D4 id="2">
<E5 id="abc" />
</D4>
</C3>
</B2>
</A1>
这可能是示例代码:
var xDoc = XDocument.Load("Test.xml");
string xPath = "//B2/C3/D4";
//or string xPath = "//B2/C3/D4[@id='1']";
var eleList = xDoc.XPathSelectElements(xPath).ToList();
foreach (var xElement in eleList)
{
Console.WriteLine(xElement);
}
它完美地工作,但是如果我向根节点A1添加命名空间,则此代码不起作用。 在搜索解决方案后,我找到this one,但它使用 Descendants()方法来查询XML。根据我的理解,如果我正在搜索&lt; E5&gt;,则此解决方案将失败。因为&lt; D7&gt;存在相同的标签,&lt; D4 id =“1”&gt;和&lt; D4 id =“2”&gt;
我的要求是搜索特定XPath中是否存在节点。如果有一种使用后代的方法,我会很高兴使用它。如果没有,请指导我如何使用名称空间进行搜索。
如果这是重复,我道歉。
答案 0 :(得分:11)
要继续使用XPath,您可以使用以下链接:
var xDoc = XDocument.Parse(@"<?xml version='1.0' encoding='utf-8'?>
<A1 xmlns='urn:sample'>
<B2>
<C3 id='1'>
<D7><E5 id='abc' /></D7>
<D4 id='1'><E5 id='abc' /></D4>
<D4 id='2'><E5 id='abc' /></D4>
</C3>
</B2>
</A1>");
// Notice this
XmlNamespaceManager nsmgr = new XmlNamespaceManager(new NameTable());
nsmgr.AddNamespace("sample", "urn:sample");
string xPath = "//sample:B2/sample:C3/sample:D4";
var eleList = xDoc.XPathSelectElements(xPath, nsmgr).ToList();
foreach (var xElement in eleList)
{
Console.WriteLine(xElement);
}
答案 1 :(得分:2)
但它使用Descendants()方法来查询XML。根据我的理解,如果我正在搜索,此解决方案将失败,因为存在相同的标记,并且
我很确定你不太了解它是如何工作的。来自MSDN documentation:
按文档顺序返回此文档或元素的后代元素的筛选集合。只有具有匹配XName的元素才会包含在集合中。
所以在你的情况下,就这样做:
xDoc.RootNode
.Descendants("E5")
.Where(n => n.Parent.Name.LocalName == "B4");
答案 2 :(得分:0)
试试这个
var xDoc = XDocument.Parse("<A1><B2><C3 id=\"1\"><D7><E5 id=\"abc\" /></D7><D4 id=\"1\"><E5 id=\"abc\" /></D4><D4 id=\"2\"><E5 id=\"abc\" /></D4></C3></B2></A1>");
foreach (XElement item in xDoc.Element("A1").Elements("B2").Elements("C3").Elements("D4"))
{
Console.WriteLine(item.Element("E5").Value);//to get the value of E5
Console.WriteLine(item.Element("E5").Attribute("id").Value);//to get the value of attribute
}