// Инициализировать Xerces и получить парсер
SAX2 XercesInitializer init;
auto_ptr<SAX2XMLReader>
parser(XMLReaderFactory::createXMLReader());
// Включить режим проверки
parser->setFeature(XMLUni::fgSAX2CoreValidation, true);
// Зарегистрировать обработчик ошибок для получения уведомлений о
// нарушениях DTD
CircusErrorHandler error;
parser->setErrorHandler(&error);
parser->parse('animals.xml');
} catch (const SAXException& e) {
cout << 'xml error ' << toNative(e.getMessage()) << '
';
return EXIT_FAILURE;
} catch (const XMLException& e) {
cout << 'xml error ' << toNative(e.getMessage()) << '
';
return EXIT_FAILURE;
} catch (const exception& e) {
cout << e.what() << '
';
return EXIT_FAILURE;
}
}
#include <exception>
#include <iostream> // cout
#include <stdexcept> // runtime_error
#include <xercesc/dom/DOM.hpp>
#include <xercesc/parsers/XercesDOMParser.hpp>
#include <xercesc/sax/HandlerBase.hpp>
#include <xercesc/util/PlatformUtils.hpp>
#include 'xerces_strings.hpp' // Пример 14.4
using namespace std;
using namespace xercesc;
/*
* Определить XercesInitializer, как это сделано в примере 14.8
* и CircusErrorHandler, как это сделано в примере 14.7
*/
int main() {
try {
// Инициализировать Xerces и сконструировать DOM-парсер.
XercesInitializer init;
XercesDOMParser parser;
// Включить режим проверки DTD
parser.setValidationScheme(XercesDOMParser::Val_Always);
// Зарегистрировать обработчик ошибок для получения уведомлений о
// нарушениях схемы
CircusErrorHandler handler;
parser.setErrorHandler(&handler);
// Выполнить синтаксический анализ вместе с проверкой.
parser.parse('animals.xml');
} catch (const SAXException& e) {
cout << 'xml error: ' << toNative(e.getMessage()) << '
';
return EXIT_FAILURE;
} catch (const XMLException& e) {
cout << 'xml error: ' << toNative(e.getMessage()) << '
';
return EXIT_FAILURE;
} catch (const exception& e) {
cout << e.what() << '
';
return EXIT_FAILURE;
}
}
Определения DTD обеспечивают простой способ наложения ограничений на документ XML. Например, в DTD можно указать, какие элементы допускаются в документе, какие атрибуты может иметь элемент и может ли конкретный элемент содержать дочерние элементы, текст или и то и другое. Можно также накладывать ограничения на тип, порядок следования и количество дочерних элементов, а также на значения атрибутов.
DTD предназначены для определения подмножества правильно сформированных документов XML, которые характерны для определенной прикладной области. Например, в примере 14.1 важно то, что каждый элемент animal
имеет дочерние элементы name
, species
, dateOfBirth
, veterinarian
и trainer
, а элементы name
, species
и dateOfBirth
содержат только текст в то время, как элементы veterinarian
и trainer
имеют атрибуты name
и phone
. Более того, элемент animal
не должен иметь атрибут phone
, а элемент veterinarian
не должен иметь дочерний элемент species
.
DTD в примере 14.11 накладывает ограничения различного типа. Например, приведенное ниже name
, species
, dateOfBirth
, veterinarian
и trainer
, задаваемых именно в этом порядке.
<!ELEMENT animal (name, species, dateOfBirth,
veterinarian, trainer) >
Аналогично приведенное ниже trainer
должен иметь атрибуты name
и phone
, а отсутствие в DTD объявлений других атрибутов для элемента дрессировщика говорит о том, что этот элемент может иметь только два атрибута.
<!ATTLIST trainer
name CDATA #REQUIRED
phone CDATA #REQUIRED
>
Документ XML, который содержит DTD и удовлетворяет его требованиям, называют SAX2XMLReader
и XercesDOMParser
не являются по умолчанию подтверждающими парсерами, в каждом из них предусмотрена функция подтверждения достоверности, которая может подключаться так, как это сделано в