C# для профессионалов. Том II - страница 4
будут создавать документ XML, который соответствует рекомендациям по пространствам имен XML 1.0 консорциума W3C.
>XmlReader
и >XmlWriter
являются абстрактными классами. Рисунок ниже показывает, какие классы являются производными от >XmlReader
и >XmlWriter
:
>XmlTextReader
и >XmlTextWriter
работают либо с объектами на основе потока, либо с объектами на основе >TextReader
или >TextWriter
. >XmlNodeReader
использует >XmlNode
вместо потока в качестве своего источника. >XmlValidatingReader
добавляет DTD и проверку схем и поэтому предлагает проверку данных. Мы рассмотрим это подробнее позже в этой главе.
XmlTextReader
>XmlTextReader
похож на SAX. Одно из различий заключается в том, что SAX является моделью типа рассылки (push), т.е. посылает данные приложению и разработчик должен быть готов принять их, a >XmlTextReader
применяет модель запроса (pull), где данные посылаются приложению, которое их запрашивает. Это предоставляет более простую и интуитивно понятную модель для программирования. Другое преимущество состоит в том, что модель запроса может быть избирательной в отношении данных, посылаемых приложению. Если нужны не все данные, то их не нужно обрабатывать. В модели рассылки все данные XML должны быть обработаны приложением, нужны они ему или нет.
Возьмем простой пример считывания данных XML, и затем более внимательно рассмотрим класс >XmlTextReader
. Код можно найти в папке >XmlReaderSample1
. Можно заменить метод >button1_Click
в предыдущем примере на следующий код. Эту версию данного кода можно найти в папке >SampleBase2
загруженного архива кода. Не забудьте изменить:
>using MSXML2;
на
>using System.Xml;
Мы должны это сделать, поскольку используем теперь не MSXML 3.0, а пространство имен >System.Xml
. Нужно также удалить метод >listBox1_SelectedIndexChanged
, так как он включает в себя некоторые неподдерживаемые методы и строку:
>private DOMDocument30 doc;
>protected void button1_Click(object sender, System.EventArgs e) {
> // Измените этот путь доступа, чтобы найти books.xml
> string fileName = "..\\..\\..\\books.xml";
> // Создать новый объект TextReader
> XmlTextReader tr = new XmlTextReader(fileName);
> // Прочитать узел за раз
> while(tr.Read()) {
> if (tr.NodeType == XmlNodeType.Text) listBox1.Items.Add(tr.Value);
> }
>}
Это >XmlTextReader
в простейшей форме. Сначала создается строковый объект >fileName
с именем файла XML. Затем создается новый объект >XmlTextReader
, передавая в качестве параметра строку >fileName.XmlTextReader
в настоящее время имеет 13 различных перегружаемых конструкторов, которые получают различные комбинации строк (имен файлов и URL), потоков и таблиц имен. После инициализации объекта >XmlTextReader
ни один узел не выбран. Это единственный момент, когда узел не является текущим. Когда мы начинаем цикл >tr.Read
, первая операция чтения >Read
переместит нас в первый узел документа. Обычно это бывает узел Declaration XML. В этом примере при переходе к каждому узлу >tr.NodeType
сравнивается с перечислением >XmlNodeType
, и когда встречается текстовый узел, значение текста добавляется в >listbox
. Вот экран после того, как было загружено окно списка:
Существует несколько способов перемещения по документу. Как мы только что видели, >Read
перемещает нас к следующему узлу. Затем можно проверить, имеет ли узел значение (>HasValue
) или, как мы скоро увидим, имеет ли узел атрибуты (>HasAttributes
). Существует метод >ReadStartElement
, который проверяет, является ли текущий узел начальным элементом, и затем перемешает текущую позицию к следующему узлу. Если текущая позиция не является начальным элементом, то порождается исключение >XmlException
. Этот метод совпадает с вызовом метода >IsStartElement
, за которым следует метод >Read
.
Методы >ReadString
и >ReadCharts
считывают текстовые данные из элемента. >ReadString
возвращает строковый объект, содержащий данные, в то время как >ReadCharts
считывает данные в заданный массив символов.
Метод >ReadElementString
аналогичен методу >ReadString
, за исключением того, что при желании можно передать в него имена элемента. Если следующий узел содержимого не является начальным тегом или, если параметр