Проверка по схеме с помощью XPathNavigator
Обновлен: November 2007
С помощью класса XmlDocument можно проверять XML-содержимое в объекте XmlDocument двумя способами. Первый способ проверки XML- содержимого заключается в использовании проверяющего объекта XmlReader, а второй — в вызове метода Validate класса XmlDocument. Можно также выполнить проверку XML-содержимого «только для чтения» с помощью класса XPathDocument.
Проверка XML-данных
Класс XmlDocument по умолчанию не выполняет проверку XML-документа с помощью определения DTD или XSD. Он только проверяет правильность формата XML-документа.
Первый способ проверки XML-документа заключается в его проверке при загрузке в объект XmlDocument с помощью проверяющего объекта XmlReader. Второй способ состоит в проверке ранее нетипизированного XML-документа с помощью метода Validate класса XmlDocument. В обоих случаях изменения в проверяемом XML-документе можно проверить повторно с помощью метода Validate класса XmlDocument.
Проверка документа при загрузке
Проверочный объект XmlReader создается путем передачи объекта XmlReaderSettings методу Create класса XmlReader, который принимает объект XmlReaderSettings в качестве параметра. Свойство ValidationType объекта XmlReaderSettings, переданного в качестве параметра, имеет значение Schema, а схема XML для XML-документа, содержащегося в объекте XmlDocument, добавляется к его свойству Schemas. После этого проверочный объект XmlReader используется для создания объекта XmlDocument.
В следующем примере выполняется проверка файла contosoBooks.xml, загруженного в объект XmlDocument путем создания объекта XmlDocument с помощью проверяющего объекта XmlReader. Поскольку XML-документ допустим в соответствии со своей схемой, ошибки или предупреждения проверки не формируются.
Imports System
Imports System.Xml
Imports System.Xml.Schema
Imports System.Xml.XPath
Class ValidatingReaderExample
Shared Sub Main(ByVal args() As String)
Try
Dim settings As XmlReaderSettings = New XmlReaderSettings()
settings.Schemas.Add("https://www.contoso.com/books", "contosoBooks.xsd")
settings.ValidationType = ValidationType.Schema
Dim reader As XmlReader = XmlReader.Create("contosoBooks.xml", settings)
Dim document As XmlDocument = New XmlDocument()
document.Load(reader)
Dim navigator As XPathNavigator = document.CreateNavigator()
Catch e As Exception
Console.WriteLine("ValidatingReaderExample.Exception: {0}", e.Message)
End Try
End Sub
Shared Sub SchemaValidationHandler(ByVal sender As Object, ByVal e As ValidationEventArgs)
Select Case e.Severity
Case XmlSeverityType.Error
Console.WriteLine("Schema Validation Error: {0}", e.Message)
Exit Sub
Case XmlSeverityType.Warning
Console.WriteLine("Schema Validation Warning: {0}", e.Message)
Exit Sub
End Select
End Sub
End Class
using System;
using System.Xml;
using System.Xml.Schema;
using System.Xml.XPath;
class ValidatingReaderExample
{
static void Main(string[] args)
{
try
{
XmlReaderSettings settings = new XmlReaderSettings();
settings.Schemas.Add("https://www.contoso.com/books", "contosoBooks.xsd");
settings.ValidationType = ValidationType.Schema;
XmlReader reader = XmlReader.Create("contosoBooks.xml", settings);
XmlDocument document = new XmlDocument();
document.Load(reader);
XPathNavigator navigator = document.CreateNavigator();
}
catch (Exception e)
{
Console.WriteLine("ValidatingReaderExample.Exception: {0}", e.Message);
}
}
static void SchemaValidationHandler(object sender, ValidationEventArgs e)
{
switch (e.Severity)
{
case XmlSeverityType.Error:
Console.WriteLine("Schema Validation Error: {0}", e.Message);
break;
case XmlSeverityType.Warning:
Console.WriteLine("Schema Validation Warning: {0}", e.Message);
break;
}
}
}
В примере в качестве входных данных используется файл contosoBooks.xml.
<bookstore xmlns="https://www.contoso.com/books">
<book genre="autobiography" publicationdate="1981-03-22" ISBN="1-861003-11-0">
<title>The Autobiography of Benjamin Franklin</title>
<author>
<first-name>Benjamin</first-name>
<last-name>Franklin</last-name>
</author>
<price>8.99</price>
</book>
<book genre="novel" publicationdate="1967-11-17" ISBN="0-201-63361-2">
<title>The Confidence Man</title>
<author>
<first-name>Herman</first-name>
<last-name>Melville</last-name>
</author>
<price>11.99</price>
</book>
<book genre="philosophy" publicationdate="1991-02-15" ISBN="1-861001-57-6">
<title>The Gorgias</title>
<author>
<name>Plato</name>
</author>
<price>9.99</price>
</book>
</bookstore>
В примере также в качестве входных данных используется contosoBooks.xsd.
<?xml version="1.0" encoding="utf-8"?>
<xs:schema attributeFormDefault="unqualified" elementFormDefault="qualified" targetNamespace="https://www.contoso.com/books" xmlns:xs="http://www.w3.org/2001/XMLSchema">
<xs:element name="bookstore">
<xs:complexType>
<xs:sequence>
<xs:element maxOccurs="unbounded" name="book">
<xs:complexType>
<xs:sequence>
<xs:element name="title" type="xs:string" />
<xs:element name="author">
<xs:complexType>
<xs:sequence>
<xs:element minOccurs="0" name="name" type="xs:string" />
<xs:element minOccurs="0" name="first-name" type="xs:string" />
<xs:element minOccurs="0" name="last-name" type="xs:string" />
</xs:sequence>
</xs:complexType>
</xs:element>
<xs:element name="price" type="xs:decimal" />
</xs:sequence>
<xs:attribute name="genre" type="xs:string" use="required" />
<xs:attribute name="publicationdate" type="xs:date" use="required" />
<xs:attribute name="ISBN" type="xs:string" use="required" />
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:schema>
В приведенном выше примере при вызове метода Load возникает исключение XmlSchemaValidationException, если какой-либо атрибут или тип элемента не соответствует типу, указанному в проверочной схеме. Если ValidationEventHandler установлен на проверяющий объект XmlReader, метод ValidationEventHandler будет вызываться для каждого недопустимого типа.
Исключение XmlSchemaException возникает, если объект XPathNavigator выполняет доступ к атрибуту или элементу, у которого свойство TypedValue имеет значение invalid.
Свойство Validity можно использовать для определения допустимости отдельного атрибута или элемента при их оценке с помощью объекта XPathNavigator.
Примечание. |
---|
Если XML-документ загружается в объект XmlDocument со связанной схемой, определяющей значения по умолчанию, объект XmlDocument рассматривает их как значения по умолчанию XML-документа. Это означает, что свойство IsEmptyElement всегда возвращает значение false для элемента, получившего значения по умолчанию из схемы, даже если в XML-документе это был пустой элемент. |
Проверка документа с помощью метода Validate
Метод Validate класса XmlDocument проверяет XML-документ, содержащийся в объекте XmlDocument, по схеме, указанной в свойстве Schemas объекта XmlDocument, и выполняет приращение информационного набора. Результатом является замена в объекте XmlDocument ранее нетипизированного XML-документа типизированным документом.
Объект XmlDocument сообщает об ошибках и предупреждениях проверки схемы с помощью делегата ValidationEventHandler, передаваемого в качестве параметра методу Validate.
Следующий пример проверяет файл contosoBooks.xml, содержащийся в объекте XmlDocument по схеме contosoBooks.xsd, содержащейся в свойстве Schemas объекта XmlDocument.
Imports System
Imports System.Xml
Imports System.Xml.Schema
Imports System.Xml.XPath
Class ValidateExample
Shared Sub Main(ByVal args() As String)
Dim document As XmlDocument = New XmlDocument()
document.Load("contosoBooks.xml")
Dim navigator As XPathNavigator = document.CreateNavigator()
document.Schemas.Add("https://www.contoso.com/books", "contosoBooks.xsd")
Dim validation As ValidationEventHandler = New ValidationEventHandler(AddressOf SchemaValidationHandler)
document.Validate(validation)
End Sub
Shared Sub SchemaValidationHandler(ByVal sender As Object, ByVal e As ValidationEventArgs)
Select Case e.Severity
Case XmlSeverityType.Error
Console.WriteLine("Schema Validation Error: {0}", e.Message)
Exit Sub
Case XmlSeverityType.Warning
Console.WriteLine("Schema Validation Warning: {0}", e.Message)
Exit Sub
End Select
End Sub
End Class
using System;
using System.Xml;
using System.Xml.Schema;
using System.Xml.XPath;
class ValidateExample
{
static void Main(string[] args)
{
XmlDocument document = new XmlDocument();
document.Load("contosoBooks.xml");
XPathNavigator navigator = document.CreateNavigator();
document.Schemas.Add("https://www.contoso.com/books", "contosoBooks.xsd");
ValidationEventHandler validation = new ValidationEventHandler(SchemaValidationHandler);
document.Validate(validation);
}
static void SchemaValidationHandler(object sender, ValidationEventArgs e)
{
switch (e.Severity)
{
case XmlSeverityType.Error:
Console.WriteLine("Schema Validation Error: {0}", e.Message);
break;
case XmlSeverityType.Warning:
Console.WriteLine("Schema Validation Warning: {0}", e.Message);
break;
}
}
}
В примере в качестве входных данных используется файл contosoBooks.xml.
<bookstore xmlns="https://www.contoso.com/books">
<book genre="autobiography" publicationdate="1981-03-22" ISBN="1-861003-11-0">
<title>The Autobiography of Benjamin Franklin</title>
<author>
<first-name>Benjamin</first-name>
<last-name>Franklin</last-name>
</author>
<price>8.99</price>
</book>
<book genre="novel" publicationdate="1967-11-17" ISBN="0-201-63361-2">
<title>The Confidence Man</title>
<author>
<first-name>Herman</first-name>
<last-name>Melville</last-name>
</author>
<price>11.99</price>
</book>
<book genre="philosophy" publicationdate="1991-02-15" ISBN="1-861001-57-6">
<title>The Gorgias</title>
<author>
<name>Plato</name>
</author>
<price>9.99</price>
</book>
</bookstore>
В примере также в качестве входных данных используется contosoBooks.xsd.
<?xml version="1.0" encoding="utf-8"?>
<xs:schema attributeFormDefault="unqualified" elementFormDefault="qualified" targetNamespace="https://www.contoso.com/books" xmlns:xs="http://www.w3.org/2001/XMLSchema">
<xs:element name="bookstore">
<xs:complexType>
<xs:sequence>
<xs:element maxOccurs="unbounded" name="book">
<xs:complexType>
<xs:sequence>
<xs:element name="title" type="xs:string" />
<xs:element name="author">
<xs:complexType>
<xs:sequence>
<xs:element minOccurs="0" name="name" type="xs:string" />
<xs:element minOccurs="0" name="first-name" type="xs:string" />
<xs:element minOccurs="0" name="last-name" type="xs:string" />
</xs:sequence>
</xs:complexType>
</xs:element>
<xs:element name="price" type="xs:decimal" />
</xs:sequence>
<xs:attribute name="genre" type="xs:string" use="required" />
<xs:attribute name="publicationdate" type="xs:date" use="required" />
<xs:attribute name="ISBN" type="xs:string" use="required" />
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:schema>
Проверка изменений
После внесения изменений в XML-документ можно проверить эти изменения по схеме XML-документа с помощью метода Validate класса XmlDocument.
В следующем примере выполняется проверка файла contosoBooks.xml, загруженного в объект XmlDocument путем создания объекта XmlDocument с помощью проверяющего объекта XmlReader. XML-документ успешно проходит проверку во время загрузки без формирования ошибок и предупреждений проверки схемы. Затем пример выполняет два изменения в XML-документе, недопустимых по схеме contosoBooks.xsd. Первое изменение вставляет недопустимый дочерний элемент, что приводит к ошибке проверки схемы. Второе изменение присваивает значение типизированному узлу, недопустимое с точки зрения типа узла, что вызывает исключение.
Imports System
Imports System.Xml
Imports System.Xml.Schema
Imports System.Xml.XPath
Class ValidatingReaderExample
Shared Sub Main(ByVal args() As String)
Try
Dim settings As XmlReaderSettings = New XmlReaderSettings()
settings.Schemas.Add("https://www.contoso.com/books", "contosoBooks.xsd")
settings.ValidationType = ValidationType.Schema
Dim reader As XmlReader = XmlReader.Create("contosoBooks.xml", settings)
Dim document As XmlDocument = New XmlDocument()
document.Load(reader)
Dim navigator As XPathNavigator = document.CreateNavigator()
Dim validation As ValidationEventHandler = New ValidationEventHandler(AddressOf SchemaValidationHandler)
navigator.MoveToChild("bookstore", "https://www.contoso.com/books")
navigator.MoveToChild("book", "https://www.contoso.com/books")
navigator.MoveToChild("author", "https://www.contoso.com/books")
navigator.AppendChild("<title>Book Title</title>")
document.Validate(validation)
navigator.MoveToParent()
navigator.MoveToChild("price", "https://www.contoso.com/books")
navigator.SetTypedValue(DateTime.Now)
Catch e As Exception
Console.WriteLine("ValidatingReaderExample.Exception: {0}", e.Message)
End Try
End Sub
Shared Sub SchemaValidationHandler(ByVal sender As Object, ByVal e As ValidationEventArgs)
Select Case e.Severity
Case XmlSeverityType.Error
Console.WriteLine("Schema Validation Error: {0}", e.Message)
Exit Sub
Case XmlSeverityType.Warning
Console.WriteLine("Schema Validation Warning: {0}", e.Message)
Exit Sub
End Select
End Sub
End Class
using System;
using System.Xml;
using System.Xml.Schema;
using System.Xml.XPath;
class ValidatingReaderExample
{
static void Main(string[] args)
{
try
{
XmlReaderSettings settings = new XmlReaderSettings();
settings.Schemas.Add("https://www.contoso.com/books", "contosoBooks.xsd");
settings.ValidationType = ValidationType.Schema;
XmlReader reader = XmlReader.Create("contosoBooks.xml", settings);
XmlDocument document = new XmlDocument();
document.Load(reader);
XPathNavigator navigator = document.CreateNavigator();
ValidationEventHandler validation = new ValidationEventHandler(SchemaValidationHandler);
navigator.MoveToChild("bookstore", "https://www.contoso.com/books");
navigator.MoveToChild("book", "https://www.contoso.com/books");
navigator.MoveToChild("author", "https://www.contoso.com/books");
navigator.AppendChild("<title>Book Title</title>");
document.Validate(validation);
navigator.MoveToParent();
navigator.MoveToChild("price", "https://www.contoso.com/books");
navigator.SetTypedValue(DateTime.Now);
}
catch (Exception e)
{
Console.WriteLine("ValidatingReaderExample.Exception: {0}", e.Message);
}
}
static void SchemaValidationHandler(object sender, ValidationEventArgs e)
{
switch (e.Severity)
{
case XmlSeverityType.Error:
Console.WriteLine("Schema Validation Error: {0}", e.Message);
break;
case XmlSeverityType.Warning:
Console.WriteLine("Schema Validation Warning: {0}", e.Message);
break;
}
}
}
В примере в качестве входных данных используется файл contosoBooks.xml.
<bookstore xmlns="https://www.contoso.com/books">
<book genre="autobiography" publicationdate="1981-03-22" ISBN="1-861003-11-0">
<title>The Autobiography of Benjamin Franklin</title>
<author>
<first-name>Benjamin</first-name>
<last-name>Franklin</last-name>
</author>
<price>8.99</price>
</book>
<book genre="novel" publicationdate="1967-11-17" ISBN="0-201-63361-2">
<title>The Confidence Man</title>
<author>
<first-name>Herman</first-name>
<last-name>Melville</last-name>
</author>
<price>11.99</price>
</book>
<book genre="philosophy" publicationdate="1991-02-15" ISBN="1-861001-57-6">
<title>The Gorgias</title>
<author>
<name>Plato</name>
</author>
<price>9.99</price>
</book>
</bookstore>
В примере также в качестве входных данных используется contosoBooks.xsd.
<?xml version="1.0" encoding="utf-8"?>
<xs:schema attributeFormDefault="unqualified" elementFormDefault="qualified" targetNamespace="https://www.contoso.com/books" xmlns:xs="http://www.w3.org/2001/XMLSchema">
<xs:element name="bookstore">
<xs:complexType>
<xs:sequence>
<xs:element maxOccurs="unbounded" name="book">
<xs:complexType>
<xs:sequence>
<xs:element name="title" type="xs:string" />
<xs:element name="author">
<xs:complexType>
<xs:sequence>
<xs:element minOccurs="0" name="name" type="xs:string" />
<xs:element minOccurs="0" name="first-name" type="xs:string" />
<xs:element minOccurs="0" name="last-name" type="xs:string" />
</xs:sequence>
</xs:complexType>
</xs:element>
<xs:element name="price" type="xs:decimal" />
</xs:sequence>
<xs:attribute name="genre" type="xs:string" use="required" />
<xs:attribute name="publicationdate" type="xs:date" use="required" />
<xs:attribute name="ISBN" type="xs:string" use="required" />
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:schema>
В приведенном выше примере в XML-документ, содержащийся в объекте XmlDocument, вносятся два изменения. Во время загрузки этого XML-документа все ошибки проверки схемы будут обрабатываться методом обработчика события проверки и выводиться на консоль.
В этом примере ошибки проверки возникли после загрузки XML-документа и были обнаружены методом Validate класса XmlDocument.
Изменения, сделанные методом SetTypedValue класса XPathNavigator, вызвали исключение InvalidCastException, поскольку новое значение недопустимо с точки зрения типа схемы узла.
Дополнительные сведения об изменении значений с помощью метода SetTypedValue см. в разделе Изменение XML-данных с помощью класса XPathNavigator.
Проверка «только для чтения»
Класс XPathDocument является доступным только для чтения, хранящимся в памяти представлением XML-документа. Классы XPathDocument и XmlDocument создают объекты XPathNavigator для перехода по XML-документам и их изменения. Поскольку класс XPathDocument доступен только для чтения, объект XPathNavigator, возвращенный из объекта XPathDocument, не может изменять XML-документ, содержащийся в объекте XPathDocument.
Для проверки можно создать объект XPathDocument точно так же, как создается объект XmlDocument, используя проверяющий объект XmlReader, как описано выше в этом разделе. Объект XPathDocument проверяет XML-документ во время загрузки, но, поскольку в объекте XPathDocument нельзя изменять XML-данные, повторно проверить этот XML-документ невозможно.
Дополнительные сведения о доступных только для чтения и изменяемых объектах XPathNavigator см. в разделе Чтение XML-данных с помощью XPathDocument и XmlDocument.
См. также
Основные понятия
Обработка XML-данных с использованием модели данных XPath
Чтение XML-данных с помощью XPathDocument и XmlDocument
Выбор, вычисление и отбор XML-данных с помощью XPathNavigator
Доступ к XML-данным с помощью класса XPathNavigator
Изменение XML-данных с помощью XPathNavigator