Поиск узлов в QDomDocument с использованием XPath

Я избалован (конечно, гнилой) C# и его классами обработки XML в пространстве имен System.Xml. Я могу загрузить файл XML в файл XmlDocument. Я могу искать по всему документу узлы, соответствующие выражению XPath, используя XmlNode.SelectNodes( "an xpath expression" ). Результатом является XmlNodeList, который содержит XmlNode объектов, которые я могу перебирать.

Сейчас я использую C++ Qt (версии 4.7.1 и 4.8, но конкретная версия может быть не важна). Я могу загрузить файл XML в файл QDomDocument. Но я разочарован тем, что не могу искать документ с помощью выражения XPath так же, как я делал это в C#.

Я имел ограниченный успех, используя QXmlQuery для поиска материала в файле XML. Если я правильно напишу запрос, я смогу получить QStringList результатов, повторить эти QStringList, а затем сохранить данные где-нибудь для последующего использования.

Но я все еще хочу иметь возможность получить коллекцию QDomNode объектов, которые находятся в документе, напрямую через выражение XPath. Один конкретный вариант использования — найти один элемент, чей атрибут «имя» имеет определенное значение, а затем заменить этот элемент новым элементом. Вот почему мне нужен сам объект QDomNode, а не просто какое-то строковое или другое представление содержимого XML, которое может предоставить QXmlQuery. Для только что упомянутого конкретного варианта использования я использую QDomElement.elementsByTagName() и итерирую эти элементы, но это не так гибко и не так круто, как XPath.

Это просто желаемое за действительное? Стоит ли разрабатывать новый класс, реализующий интерфейс QAbstractXmlReceiver? Или я просто получу новую коллекцию данных, которая не имеет прямого отношения к объектам QDomNode в QDomDocument?


person Mike Finch    schedule 09.05.2019    source источник
comment
Очень странно, что команда Qt решила сохранить отдельные модели данных для DOM и XQuery/XPath/XSLT, но они не восполнили пробел какой-либо готовой реализацией QAbstractXmlNodeModel. Вы можете найти его по адресу adared.ch/qdomnodemodel-qxmlquery.   -  person Alejandro    schedule 09.05.2019
comment
Спасибо, @Алехандро. Это было отличное предложение. Исходный код QDomNodeModel доступен по этой ссылке в сочетании с примером использования QXmlQuery по адресу qtcentre.org/threads/ достаточно развит, чтобы работать у меня как есть.   -  person Mike Finch    schedule 10.05.2019


Ответы (1)


Ниже приведена служебная функция, которую я использую для поиска узлов в QDomDocument с использованием выражения XPath. Он использует класс QDomNodeModel, предложенный @Alejandro, который можно загрузить с https://adared.ch/qdomnodemodel-qxmlquery. Он основан на примере использования из https://www.qtcentre.org/threads/37645-QAbstractXmlNodeModel-implementation-QDomNodeModel-QXmlQuery. Спасибо Станиславу Адашевски, который предоставил как класс QDomNodeModel, так и пример использования.

В QDomNodeModel есть несколько методов, которые закомментированы как нереализованные. Однако для простого содержимого XML, которое мне нужно было найти, QDomNodeModel достаточно как есть.

//
/// @brief Search for nodes in a QDomDocument using an XPath.
/// @note I cannot return a QDomNodeList, because it has no public methods for adding items to it.
/// @param[in] doc The document to search.
/// @param[in] fromNode The node in the document to start searching from.
///   e.g., to search the whole document, use <code>doc.documentElement()</code>.
/// @param[in] xpath The XPath expression.
/// @return A list of found nodes.
//
QList<QDomNode> findNodes( QDomDocument const & doc, QDomNode const & fromNode, QString const & xpath )
{
  qDebug( "%s", __FUNCTION__ );
  QList<QDomNode> foundNodes;

  //------------------------------
  // The name pool that everybody shares.
  QXmlNamePool pool;

  //------------------------------
  // The model that wraps the document.
  QDomNodeModel model( pool, doc );

  //------------------------------
  // The query.
  // XQuery10 means the default XQuery 1.0 language, as opposed to XSLT20.
  QXmlQuery query( /*QXmlQuery::XQuery10,*/ pool );

  // Operate on the given node.
  QXmlNodeModelIndex fromIndex = model.fromDomNode( fromNode );
  query.setFocus( QXmlItem( fromIndex ) );

  // The query statement.
  query.setQuery( xpath );
  if ( !query.isValid() )
  {
    qDebug( "Query is not valid" );
    return foundNodes;
  }

  //------------------------------
  // The destination for the result of the query.
  QXmlResultItems result;

  //------------------------------
  // Evaluate the query.
  query.evaluateTo( &result );
  if ( result.hasError() )
  {
    qDebug( "Query evaluation failed" );
    return foundNodes;
  }

  //------------------------------
  // The result of the query.
  qDebug( "Query result:" );
  while ( !result.next().isNull() )
  {
    QXmlNodeModelIndex index = result.current().toNodeModelIndex();
    QDomNode node = model.toDomNode( index );
    qDebug( "  %d %s: %s", node.nodeType(), qPrintable( node.nodeName() ), qPrintable( node.nodeValue() ) );
    foundNodes << node;
  }

  return foundNodes;
}

В моем приложении я загружаю XML-файл и использую указанную выше служебную функцию для его поиска.

//------------------------------
// The path of the XML file.
QString path = "settings.xml";

//------------------------------
// Open the file.
QFile file( path );
if ( !file.open( QIODevice::ReadOnly ) )
{
  qDebug( "Failed to open '%s': %s", qPrintable( path ), qPrintable( file.errorString() ) );
  return;
}

//------------------------------
// Load the file into a document.
QDomDocument doc;
QString error;
int line;
int column;
if ( !doc.setContent( &file, &error, &line, &column ) )
{
  qDebug( "%s(%d,%d): %s", qPrintable( path ), line, column, qPrintable( error ) );
  return;
}

//------------------------------
// The document root element.
QDomElement rootElem = doc.documentElement();

//------------------------------
// Search for an element whose name attribute has a certain value.
QString name = "Alice";
QString xpath = QString( "setting[@name='%1']" ).arg( name );
QList<QDomNode> foundNodes = findNodes( doc, rootElem, xpath );

//------------------------------
// Did I find it?
if ( foundNodes.size() > 0 )
{
  QDomElement foundElem = foundNodes.at( 0 ).toElement();

  // Do something with that element.      
  ...
} 

Пример содержимого XML для поиска.

<?xml version='1.0'?>
<settings>
  <setting name="Bob">12</setting>
  <setting name="Carol">34</setting>
  <setting name="Ted">56</setting>
  <setting name="Alice">78</setting>
</settings>
person Mike Finch    schedule 15.05.2019