Используйте 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 4

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>