memStrm.Seek(0, SeekOrigin, Begin);
// читаем из потока в памяти в объект XmlDocument
doc.load(strmRead);
// получить все элементы продуктов
XmlNodeList nodeLst=doc.GetElementsByTagName('ProductName');
// загрузить их в окно списка
foreach(XmlNode nd in nodeLst) listBox1.Items.Add(nd.InnerText);
}
private void listBox1_SelectedIndexChanged(object sender, System.EventArgs e) {
// при щелчке в окне списка
// появляется окно сообщения с ценой изделия
string srch=
'XmlProducts/products[ProductName= ' + ''' + listBox1.SelectedItem.ToString() + ']';
XmlNode foundNode=doc.SelectSingleNode(srch);
if (foundNode!=null)
MessageBox.Show(foundNode.SelectSingleNode('UnitPrice').InnerText);
else MessageBox.Show('Not found');
}
На следующем экране можно видеть данные в списке, а также в таблице данных:
Если желательно сохранить документ XML на диске, то нужно сделать примерно следующее:
string file = 'с:\test\product.xml';
ds.WriteXml(file);
Это даст нам правильно сформированный документ XML на диске, который можно прочитать посредством другого потока, с помощью DataSet, или может использоваться другим приложением или web-сайтом. Так как никакого параметра XmlMode не определено, этот документ XmlDocument будет содержать схему. В нашем примере в качестве параметра для метода XmlDocument.Load используется поток.
Когда XmlDocument подготовлен, мы загружаем listbox с помощью того же объекта XPath, который использовался раньше. Если посмотреть внимательно, то можно заметить, что слегка изменено событие listBox1_SelectedIndexChanged. Вместо вывода InnerText элемента, выполняется другой поиск XPath с помощью SelectSingleNode, чтобы получить элемент UnitPrice. Каждый раз при щелчке на продукте в listbox будет появляться MessageBox для UnitPrise. Теперь у нас есть два представления данных, но более важно то, что имеется возможность манипулировать данными с помощью двух различных моделей. Можно использовать пространство имен Data для данных или пространство имен XML через данные. Такой подход ведет к очень гибким конструкциям в приложениях, так как теперь при программировании нет жесткой связи только с одной объектной моделью. Таким образом, мы имеем несколько представлений одних и тех же данных и несколько способов доступа к данным.
Следующий пример будет упрощать процесс, удаляя три потока и используя некоторые возможности ADO, встроенные в пространство имен XML. Нам понадобится изменить строку кода на уровне модуля:
private XmlDocument doc=new XmlDocument();
на:
private XmlDataDocument doc;
Это нужно сделать, так как мы не собираемся использовать XmlDataDocument. Вот код, который можно найти в папке ADOSample2:
private void button1_Click(object sender, System.EventArgs e) {
// создать множество данных (DataSet)
DataSet ds=new DataSet('XMLProducts');
// соединиться с базой данных northwind и
//выбрать все строки из таблицы products
//выполнить изменения в строке подключения с учетом имени пользователя и имени сервера
SqlConnection conn=
new SqlConnection(@'server=GLYNNJ_CS NetSDK;uid=sa;pwd=;database=northwind');
SqlDataAdapter da=new SqlDataAdapter('select * from products', conn);
// заполнить множество данных
da.Fill(ds, 'products');
// загрузить данные в сетку
dataGrid1.DataSource=ds;
dataGrid1.DataMember='products';
doc=new XmlDataDocument(ds);
// извлечь все элементы продуктов
XmlNodeList nodeLst=doc.GetElementsByTagName('ProductName');
// загрузить их в окно списка
// здесь используется цикл for
for(int ctr=0; ctr<nodeLst.Count; ctr++) listBox1.Items.Add(nodeLst [ctr].InnerText);
}
Как можно видеть, код для загрузки DataSet в документ XML был упрощен. Вместо использования класса XmlDocument, используется класс XmlDataDocument. Этот класс был создан специально для использования данных с объектом DataSet.
XmlDataDocument базируется на классе XmlDocument, поэтому он имеет всю функциональность класса XmlDocument. Одним из основных отличий является перегруженный конструктор для XmlDataDocument. Отметим строку кода, где создается экземпляр XmlDataDocument:
XmlDataDocument doc=new XmlDataDocument(ds);
Он передает в качестве параметра созданный объект DataSet, ds. Документ XML создается из множества данных, поэтому не требуется использование метода Load. Существует также свойство DataSet, которое может задаваться с помощью текущего свойства DataSet. Фактически, если создается новый объект XmlDataDocument без передачи DataSet в качестве параметра, то он содержит объект DataSet с именем NewDataSet, который не имеет DataTables в коллекции таблиц. Существует также свойство DataSet, которое можно установить после создания объекта на основе XmlDataDocument. Если после вызова DataSet.Fill добавляется следующая строка кода:
ds.WriteXml('с:\test\sample.xml' , XmlWriteMode, WriteSchema);
…создается следующий XML. Отметим, что мы включили в документ схему XSD. Если нежелательно, чтобы схема включалась в файл, то можно передать член перечисления XmlWriteMode.IgnoreSchema:
<?xml version='1.0' standalone='yes'?>
<XMLProducts>
<xsd:schema id='XMLProducts' targetNamespace='' xmlns='' xmlns:xsd='http://www.w3.org/2001/XMLSchema' xmlns:msdata='urn:schemas-microsoft-com:xml-msdata'>
