как игнорировать пространства имен с помощью XPath

111

Моя цель - извлечь определенные узлы из нескольких файлов xml с несколькими пространствами имен с помощью XPath. Все работает нормально, пока я знаю URI пространства имен. Само имя пространства имен остается постоянным, но схемы (XSD) иногда создаются клиентом, т.е. мне неизвестны. Тогда у меня остается три варианта:

  1. используйте только одну схему для пространства имен, надеясь, что ничего не пойдет не так (могу ли я быть уверен?)

  2. получите дочерние узлы документа и найдите первый узел с URI пространства имен, надеясь, что он там, и просто используйте URI, надеясь, что он правильный. может пойти не так по нескольким причинам

  3. как-то сказать xpath: «послушайте, мне плевать на пространства имен, просто найдите ВСЕ узлы с этим именем, я могу даже сказать вам имя пространства имен, только не URI». И вот в чем вопрос ...

Это не повторение многочисленных вопросов типа «мое выражение xpath не работает, потому что я не знаю об осведомленности о пространстве имен», которые можно найти здесь или здесь . Я знаю, как использовать осведомленность о пространстве имен. Только не как от этого избавиться.

Костя
источник
2
Если вы не знаете схемы, как узнать, какие элементы вам нужны?
Пол Бутчер,
4
Точная
1
спасибо за указание, Алехандро. Поиск по запросу «ignore namespace xpath» должен был выявить это, но этого не
произошло
2
@kostja: Не ищите с таким окном поиска, это бесполезно ... Попробуйте Google в следующий раз. Фактически, это поощряется командой SO.
1
Google sitesearch действительно лучше справляется с поиском полезных вещей на SO. Интересно, почему это не вариант по умолчанию. Еще раз спасибо, Алехандро
Костя

Ответы:

166

Вы можете использовать local-name()функцию XPath. Вместо выбора узла вроде

/path/to/x:somenode

вы можете выбрать все узлы и отфильтровать узел с правильным локальным именем:

/path/to/*[local-name() = 'somenode']
Дирк Фольмар
источник
9
Вы также можете использовать local-name()для ссылки на атрибуты, не зная о пространстве имен, см .: stackoverflow.com/q/21239181/274677
Marcus Junius Brutus
Взгляните на этот учебник: codeimplify.com/java/java-xpath-ignore-namespace-example
hipokito
1
Так просто. спас мой день.
C Johnson
21

Вы можете сделать то же самое в XPath2.0 с менее подробным синтаксисом:

/path/to/*:somenode
Андрес Куадрос Суарес
источник
3

Вы можете использовать Namespace = false в XmlTextReader

[TestMethod]
public void MyTestMethod()
{
    string _withXmlns = @"<?xml version=""1.0"" encoding=""utf-8""?>
<ParentTag xmlns=""http://anyNamespace.com"">
<Identification value=""ID123456"" />
</ParentTag>
";

    var xmlReader = new XmlTextReader(new MemoryStream(Encoding.Default.GetBytes(_withXmlns)));

    xmlReader.Namespaces = false;

    var content = XElement.Load(xmlReader);

    XElement elem = content.XPathSelectElement("/Identification");

    elem.Should().NotBeNull();
    elem.Attribute("value").Value.Should().Be("ID123456");
}

с участием :

using System;
using System.IO;
using System.Linq;
using System.Text;
using System.Xml;
using System.Xml.Linq;
using System.Xml.XPath;
using FluentAssertions;
using Microsoft.VisualStudio.TestTools.UnitTesting;
Пьер Вондершер
источник
Для выбора узла через XPath это работает; К сожалению, вы не можете сохранить документ из-за 'The 'xmlns' attribute is bound to the reserved namespaceошибки.
AutomatedChaos
2

Или вы можете использовать name ():

/path/to/*[name() = 'somenode']

Или только поисковые атрибуты:

//*[@attribute="this one"]

Если вы открываете xml как объект powershell, он игнорирует пространства имен:

[xml]$xml = get-content file.xml
$xml.path.to.somenode
js2010
источник
0

Это мой пример в Qt C ++. Qt поддерживает XPath 2.0:

    QString planePath = ":/Models/Plane.dae";
    QFile f(planePath);
    if (!f.open(QIODevice::ReadOnly))
    {
        std::cerr << "Failed to load the file: " <<
                     planePath.toStdString() << std::endl;
        return;
    }

    QXmlQuery query;
    query.bindVariable("myFile", &f);
//    query.setQuery("doc($myFile)//*[local-name() = 'p']/text()"); // it works too but it is XPath 1.0
    query.setQuery("doc($myFile)//*:p/text()");

    QString result;
    query.evaluateTo(&result);
    qDebug() << result;
    f.close();

Вывод программы: "1 0 0 2 0 1 0 0 2 1 0 3 3 0 4 2 0 5\n"

Plane.dae

<?xml version="1.0" encoding="utf-8"?>
<COLLADA xmlns="http://www.collada.org/2005/11/COLLADASchema" version="1.4.1" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
  <asset>
    <contributor>
      <author>Blender User</author>
      <authoring_tool>Blender 2.83.3 commit date:2020-07-22, commit time:06:01, hash:353e5bd7493e</authoring_tool>
    </contributor>
    <created>2020-08-03T14:03:19</created>
    <modified>2020-08-03T14:03:19</modified>
    <unit name="meter" meter="1"/>
    <up_axis>Z_UP</up_axis>
  </asset>
  <library_effects>
    <effect id="PlaneMaterial-effect">
      <profile_COMMON>
        <technique sid="common">
          <lambert>
            <emission>
              <color sid="emission">0 0 0 1</color>
            </emission>
            <diffuse>
              <color sid="diffuse">0.01664001 0.8000001 0.01191879 1</color>
            </diffuse>
            <reflectivity>
              <float sid="specular">0.5</float>
            </reflectivity>
          </lambert>
        </technique>
      </profile_COMMON>
    </effect>
  </library_effects>
  <library_images/>
  <library_materials>
    <material id="PlaneMaterial-material" name="PlaneMaterial">
      <instance_effect url="#PlaneMaterial-effect"/>
    </material>
  </library_materials>
  <library_geometries>
    <geometry id="Plane-mesh" name="Plane">
      <mesh>
        <source id="Plane-mesh-positions">
          <float_array id="Plane-mesh-positions-array" count="12">-1 -1 0 1 -1 0 -1 1 0 1 1 0</float_array>
          <technique_common>
            <accessor source="#Plane-mesh-positions-array" count="4" stride="3">
              <param name="X" type="float"/>
              <param name="Y" type="float"/>
              <param name="Z" type="float"/>
            </accessor>
          </technique_common>
        </source>
        <source id="Plane-mesh-normals">
          <float_array id="Plane-mesh-normals-array" count="3">0 0 1</float_array>
          <technique_common>
            <accessor source="#Plane-mesh-normals-array" count="1" stride="3">
              <param name="X" type="float"/>
              <param name="Y" type="float"/>
              <param name="Z" type="float"/>
            </accessor>
          </technique_common>
        </source>
        <source id="Plane-mesh-map-0">
          <float_array id="Plane-mesh-map-0-array" count="12">1 0 0 1 0 0 1 0 1 1 0 1</float_array>
          <technique_common>
            <accessor source="#Plane-mesh-map-0-array" count="6" stride="2">
              <param name="S" type="float"/>
              <param name="T" type="float"/>
            </accessor>
          </technique_common>
        </source>
        <vertices id="Plane-mesh-vertices">
          <input semantic="POSITION" source="#Plane-mesh-positions"/>
        </vertices>
        <triangles material="PlaneMaterial-material" count="2">
          <input semantic="VERTEX" source="#Plane-mesh-vertices" offset="0"/>
          <input semantic="NORMAL" source="#Plane-mesh-normals" offset="1"/>
          <input semantic="TEXCOORD" source="#Plane-mesh-map-0" offset="2" set="0"/>
          <p>1 0 0 2 0 1 0 0 2 1 0 3 3 0 4 2 0 5</p>
        </triangles>
      </mesh>
    </geometry>
  </library_geometries>
  <library_visual_scenes>
    <visual_scene id="Scene" name="Scene">
      <node id="Plane" name="Plane" type="NODE">
        <matrix sid="transform">1 0 0 0 0 1 0 0 0 0 1 0 0 0 0 1</matrix>
        <instance_geometry url="#Plane-mesh" name="Plane">
          <bind_material>
            <technique_common>
              <instance_material symbol="PlaneMaterial-material" target="#PlaneMaterial-material">
                <bind_vertex_input semantic="UVMap" input_semantic="TEXCOORD" input_set="0"/>
              </instance_material>
            </technique_common>
          </bind_material>
        </instance_geometry>
      </node>
    </visual_scene>
  </library_visual_scenes>
  <scene>
    <instance_visual_scene url="#Scene"/>
  </scene>
</COLLADA>
8Observer8
источник