我有一个类似以下的XML文件:
<Main>
<Element1 Property1="Hello" Property3="world">
<Element1.Property2>again</Element1.Property2>
<Element1.Property3>Dave</Element1.Property3>
</Element1>
<Element2 Property1="Hello" Property3="world">
<Element2.Property2>again</Element2.Property2>
<Element2.Property1>
<SpecialElementForSayingHi/>
</Element2.Property1>
</Element2>
</Main>
我需要使用XPath匹配以下元素 - 除非有一种方法可以使用模式禁止它们存在,但我不相信存在:
<Element1.Property3>Dave</Element1.Property3>
...
<Element2.Property1>
<SpecialElementForSayingHi/>
</Element2.Property1>
具体来说,我需要匹配元素名称格式为:
的所有元素ParentElementName.NameOfAttributeThatExistsOnTheParentElement
我在.Net工作,宁愿不使用外部库,所以如果可以使用XPath 1.0来实现这一点,那将是理想的选择。如果它的效率要高得多,我愿意选择匹配重复属性而不是元素的系统。
编辑:实际上没有问题。我该怎么做?
答案 0 :(得分:2)
您可以使用LINQ2XML
XElement doc=XElement.Load("yourXML.xml");
foreach(XElement elm in doc.Elements())
{
var attr=elm.Attributes();
foreach(var elm1 in elm.Elements())
{
if(attr.Any(x=>elm1.Name.ToString().EndsWith(x.Name.ToString()))
elm1.ToString();//this contains your required XML
}
}
答案 1 :(得分:2)
我尝试使用XPAth 1.0但没有成功,也许可以使用XPath 2或XQuery。 在.NET中,最好使用LINQ:
var xml = @"
<Main>
<Element1 Property1=""Hello"" Property3=""world"">
<Element1.Property2>again</Element1.Property2>
<Element1.Property3>Dave</Element1.Property3>
</Element1>
<Element2 Property1=""Hello"" Property3=""world"">
<Element2.Property2>again</Element2.Property2>
<Element2.Property1>
<SpecialElementForSayingHi/>
</Element2.Property1>
</Element2>
</Main>";
var doc = XDocument.Load(new StringReader(xml));
var result = doc.Root.Descendants().
Where(x => x.Parent.Attributes().
Any(y => x.Name == x.Parent.Name + "." + y.Name)
);
如果你想获得字符串作为结果,你可以这样做
var result = doc.Root.Descendants().
Where(x => x.Parent.Attributes().
Any(y => x.Name == x.Parent.Name + "." + y.Name)
).Select(e => e.ToString()).Aggregate((current, next) => current + next);
答案 2 :(得分:2)
这不可能与XPath 1.0一起使用,因为它缺少范围变量等功能或者将函数指定为位置步骤的功能。
这是一个XPath 2.0解决方案:
/*/*/*
[substring-before(name(), '.') eq name(..)
and
substring-after(name(), '.') = ../@*/name()
]
基于XSLT 2.0的验证:
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:template match="/">
<xsl:sequence select=
"/*/*/*
[substring-before(name(), '.') eq name(..)
and
substring-after(name(), '.') = ../@*/name()
]"/>
</xsl:template>
</xsl:stylesheet>
对提供的XML文档应用此转换时:
<Main>
<Element1 Property1="Hello" Property3="world">
<Element1.Property2>again</Element1.Property2>
<Element1.Property3>Dave</Element1.Property3>
</Element1>
<Element2 Property1="Hello" Property3="world">
<Element2.Property2>again</Element2.Property2>
<Element2.Property1>
<SpecialElementForSayingHi/>
</Element2.Property1>
</Element2>
</Main>
评估XPath表达式并将此评估结果复制到输出中:
<Element1.Property3>Dave</Element1.Property3>
<Element2.Property1>
<SpecialElementForSayingHi/>
</Element2.Property1>