C# для профессионалов. Том II
Шрифт:
}
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">
<xsd:element name="XMLProducts" msdata:IsDataSet="true">
<xsd:complexType>
<xsd:choice maxOccurs="unbounded">
<xsd:element name="products">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="ProductID" type="xsd:int" minOccurs="0" />
<xsd:element name="ProductName" type="xsd:string" minOccurs="0" />
<xsd:element name="SupplierID" type="xsd:int" minOccurs ="0" />
<xsd:element name="CategoryID" type="xsd:int" minOccurs="0" />
<xsd:element name="QuantityPerUnit" type="xsd:string" minOccurs="0" />
<xsd:element name="UnitPrice" type="xsd:decimal" minOccurs="0" />
<xsd:element name="UnitsInStock" type="xsd:short" minOccurs="0" />
Поделиться с друзьями: