Используйте XPath для сопоставления элементов, имена которых находятся в формате ParentElement.Свойство, когда свойство уже объявлено в качестве атрибута родительского объекта
У меня есть 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, это было бы идеально. Если это так значительно более эффективным я был бы готов пойти с системой, которая соответствует дублирующему атрибуту вместо элемента.
EDIT: на самом деле не было вопроса. Как мне это сделать?
3 ответа:
Я пытался сделать это с 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);
Вы можете использовать
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 } }
Это невозможно сделать с 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>