Решение
Используйте библиотеку Xalan. Во-первых, выполните синтаксический анализ документа XML и получите указатель на xalanc::XalanDocument
. Это можно сделать, используя экземпляры XalanSourceTreeInit
, XalanSourceTreeDOMSupport
и XalanSourceTreeParserLiaison
, каждый из которых следующим образом определяется в пространстве имен xalanc
.
#include
#include
#include
#include
...
int main() {
...
// Инициализировать подсистему XalanSourceTree
XalarSourceTreeInit init;
XalanSourceTreeDOMSupport support;
// Интерфейс доступа к парсеру
XalanSourceTreeParserLiaison liaison(support);
// Подключить DOMSupport к ParserLiaison
support.setParserLiaison(&liaison);
LocalFileInputSource src(место-расположения-документа);
XalanDocument* doc = liaison.ParseXMLStream(doc);
...
}
Можно поступить по-другому и использовать парсер Xerces DOM для получения указателя на DOMDocument
, как это сделано в примере 14.14, и затем использовать экземпляры XercesDOMSupport
, XercesParserLiaison
и XercesDOMWrapperParsedSource
, каждый из которых определяется в пространстве имен xalanc
для получения указателя на XalanDocument
, соответствующего документу DOMDocument
.
#include
#include
#include
#include
...
int main() {
...
DOMDocument* doc = ...;
XercesDOMSupport support;
XercesParserLiaison liaison(support);
XercesDOMWrapperParsedSource src(doc, liaison, support);
XalanDocument* xalanDoc = src.getDocument();
...
}
На следующем шаге получите указатель на узел, выполняющий роль узла контекста при вычислении выражения XPath. Это можно сделать с помощью интерфейса DOM документа XalanDocument
. Сконструируйте XPathEvaluator
для вычисления выражения XPath и XalanDocumentPrefixResolver
для разрешения префиксов пространств имен в документе XML. Наконец, вызовите метод XPathEvaluator::evaluate()
, передавая в качестве аргументов DOMSupport
, контекстный узел, XPath-выражение и PrefixResolver
. Результат вычисления выражения возвращается в виде объекта типа XObjectPtr
; тип допустимых операций над этим объектом зависит от типа его данных XPath, который можно узнать при помощи метода getType()
.
Например, пусть требуется извлечь список имен животных из документа animals.xml , представленного в примере 14.1. Вы можете это сделать, выполняя синтаксический анализ документа и вычисляя XPath-выражение animalList/animal/name/child::text()
с использованием корня документа в качестве контекстного узла. Это проиллюстрировано в примере 14.23.
Пример 14.23. Вычисление ХРаth-выражения, используя Xalan
#include // size_t
#include
#include // cout
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "animal.hpp"
#include "xerces_strings.hpp"
using namespace std;
using namespace xercesc;
using namespace xalanc;
// Утилита RAII, которая инициализирует парсер и процессор XPath, освобождая
// ресурсы при выходе из области видимости
class XPathInitializer {
public:
XPathInitializer() {
XMLPlatformUtils::Initialize();
XPathEvaluator::initialize();
}
~XPathInitializer() {
XpathEvaluator::terminate();
XMLPlatformUtils::Terminate();
}
private:
// Запретить копирование и присваивание
XPathInitializer(const XPathInitializer&);
XPathInitializer& operator=(const XPathInitializer&);
};
// Получает уведомления об ошибках
class CircusErrorHandler : public DefaultHandler {
public:
void error(const SAXParseException& e) {
throw runtime_error(toNative(e.getMessage()));
}
void fatalError(const SAXParseException& e) { error(e); }
};
int main() {
try {
// Инициализировать Xerces и XPath и сконструировать парсер DOM.
XPathInitializer init;
XercesDOMParser parser;
// Зарегистрировать обработчик ошибок
CircusErrorHandler error;
parser.setErrorHandler(&error);
// Выполнить синтаксический анализ animals.xml.
parser.parse(fromNative("animals.xml").c_str());
DOMDocument* doc = parser.getDocument();
DOMElement* animalList = doc->getDocumentElement();
// Создать XalanDocument на основе doc.
XercesDOMSupport support;
XercesParserLiaison liaison(support);
XercesDOMWrapperParsedSource src(doc, liaison, support);
XalanDocument* xalanDoc = src.getDocument();
// Вычислить XPath-выражение для получения списка
// текстовых узлов, содержащих имена животных
XPathEvaluator evaluator;
XalanDocumentPrefixResolver resolver(xalanDoc);
XercesString xpath =
fromNative("animalList/animal/name/child::text()");
Читать дальше