Опубликован: 05.08.2010 | Уровень: специалист | Доступ: свободно
Самостоятельная работа 6:

Редактирование данных OLE DB средствами ADO.NET

Упражнение 4. Сохранение данных БД в Xml-файлах

Xml-формат является универсальным на любых платформах. Он не зависит от специфики формата БД. Поэтому его удобно применять данными между различными БД. Еще одно достоинство - текстовое представление данных. Это очень удобно при пересылке данных по сети, поскольку не содержит в себе случайных управляющих символов и сочетаний байт, которые могут быть восприняты в процессе передачи как служебные команды.

На фрагментах кода поясним работу приложения для данного упражнения.

Прежде всего сформируем строку соединения с помощью библиотечного объекта OleDbConnectionStringBuilder, которую возвращает функция

// Строка соединения к БД с абсолютным путем, определяемым сборкой
        String ConnectionString()
        {
            // Используем построитель строки подключения 
            OleDbConnectionStringBuilder objConnectionStringBuilder =
                new OleDbConnectionStringBuilder();
            objConnectionStringBuilder.Provider = "Microsoft.Jet.OLEDB.4.0";
            objConnectionStringBuilder.DataSource =
                Application.StartupPath.ToString() + @"\Data\Northwind.mdb";
            // Текущий путь
            //System.Environment.CurrentDirectory.ToString()
            // Путь к сборке + имя сборки с расширением .exe
            //Application.ExecutablePath + @"\Data\Northwind.mdb";
    
            return objConnectionStringBuilder.ToString();
        }
Далее мы создаем метод, который возвращает заполненный набор
  данных. 

  
    
    
        DataSet LoadDataSet()
        {
            DataSet ds= new DataSet();
            
            // Заполняем множественный набор данных из БД
            using (OleDbConnection conn = new OleDbConnection(ConnectionString()))
            {
                OleDbCommand selectCommand = conn.CreateCommand();// Команда получила соединение
                OleDbDataAdapter adapter = new OleDbDataAdapter(selectCommand);
                //adapter.FillSchema(ds, SchemaType.Source);// Лишнее, загружается вместе с данными
    
                selectCommand.CommandText = "SELECT * FROM Employees";
                // Загружает данные и схему (по умолчанию) первой таблицы
                // Без маркера может добавиться только одна
                // таблица и будет иметь дежурное имя Table
                adapter.Fill(ds, "Empl");
    
                // Загружает данные и схему второй таблицы
                selectCommand.CommandText = "SELECT * FROM Products";
                adapter.Fill(ds, "Prod");
                // Чтобы повторно не добавлялось одно и то же 
                adapter.MissingSchemaAction = MissingSchemaAction.AddWithKey;
                adapter.Fill(ds, "Prod");// Не добавится, т.к. установили AddWithKey
    
                //adapter.FillSchema(ds, SchemaType.Source);// Загрузить только схему
            }
    
            return ds;
        }

Обратите внимание, что в один и тот же объект DataSet загружаются две таблицы сразу. Причем сделана попытка вторую таблицу загрузить дважды. Этому препятствует свойство адаптера MissingSchemaAction, установленное в значение одноименного перечисления MissingSchemaAction.AddWithKey. Если в методе Fill() адаптера не указать имя добавляемой таблицы, то будет загружена первая таблица с дежурным именем Table. Имя загруженной таблицы можно присвоить и после ее загрузки, только тогда может успешно загрузится и вторая таблица

DataSet LoadDataSet()
        {
            DataSet ds= new DataSet();
            
            // Заполняем множественный набор данных из БД
            using (OleDbConnection conn = new OleDbConnection(ConnectionString()))
            {
                OleDbCommand selectCommand = conn.CreateCommand();// Команда получила соединение
                OleDbDataAdapter adapter = new OleDbDataAdapter(selectCommand);
    
                // Загружает данные и схему первой таблицы
                selectCommand.CommandText = "SELECT * FROM Employees";
                adapter.Fill(ds);
                ds.Tables[0].TableName="Empl";

                // Загружает данные и схему второй таблицы
                selectCommand.CommandText = "SELECT * FROM Products";
                adapter.Fill(ds);
                ds.Tables[1].TableName = "Prod";
            }
    
            return ds;
        }

В коде клиента заполненный множественный набор данных можно отобразить, связав его с созданными объектами сетки любым из двух равнозначных способов

// Заполняем множественный набор данными из БД
            DataSet ds = LoadDataSet();
    
            // Связываем с сетками
            // Первый способ
            dataGridView1.DataSource = ds.Tables["Empl"];// Как таблицу 
            // Второй способ
            dataGridView2.DataSource = ds;// Как множественный набор
            dataGridView2.DataMember = "Prod";// Уточняем из набора

Набор данных может сохранять загруженные в него объектные таблицы в файл формата Xml с помощью своего метода WriteXml. Точно такой же метод имеет и объект таблицы. Существенной разницы между работой этих методов нет, оба они могут применяться с дополнительным параметром перечисления XmlWriteMode, который может указывать своим значением XmlWriteMode.WriteSchema, чтобы в Xml-файл включалась и схема таблиц. Эти объекты имеют и симметричные методы DataSet.ReadXml и DataTable.ReadXml. Но здесь есть разница, для работы метода DataSet.ReadXml не требуется Xml-файл со схемой.

// Заполняем множественный набор данными из БД
            DataSet ds = LoadDataSet();
    
            // Путь к каталогу записи
            String path = Application.StartupPath.ToString() + @"\Data\";
            // Записываем Xml-файл без схемы
            //ds.WriteXml(path + "ds.xml", XmlWriteMode.WriteSchema);
            ds.WriteXml(path + "ds.xml");// Без схемы
    
            // Извлекаем данные из Xml-файла с помощью DataSet и показываем
            ds = new DataSet();// Создаем новый объект, используем ту же ссылку
            ds.ReadXml(path + "ds.xml");// Загружаем из Xml-файла
            // Привязываем наборы данных к сеткам для отображения
            dataGridView1.DataSource = ds.Tables["Empl"];// Как таблицу 
            dataGridView2.DataSource = ds;// Как множественный набор
            dataGridView2.DataMember = "Prod";// Уточняем из набора
    
            // Удаляем файл (просто так, для примера)
            System.IO.File.Delete(path + "ds.xml");

А для метода DataTable.ReadXml требуется файл со схемой

// Заполняем множественный набор данными из БД
            DataSet ds = LoadDataSet();
            // Сохраняем физически таблицу из набора в другой таблице 
            DataTable table = ds.Tables["Empl"].Copy();
            // Путь к каталогу записи
            String path = Application.StartupPath.ToString() + @"\Data\";
            // Сохраняем таблицу со схемой в файле Xml
            table.WriteXml(path + "Employees.xml", XmlWriteMode.WriteSchema);
            //table.WriteXml(path + "Employees.xml"); // Сохраняем без схемы
            table = null;// Бросаем объект
            table = new DataTable();// Создаем новый объект таблицы
            // Читаем таблицу из Xml-файла
            table.ReadXml(path + "Employees.xml");
            // Показываем таблицу пользователю
            dataGridView.DataSource = table;

Но прочитать файл без схемы можно с помощью DataSet.ReadXml

// Заполняем множественный набор данными из БД
            DataSet ds = LoadDataSet();
            // Сохраняем физически таблицу из набора в другой таблице 
            DataTable table = ds.Tables["Empl"].Copy();
            // Путь к каталогу записи
            String path = Application.StartupPath.ToString() + @"\Data\";
            // Сохраняем таблицу со схемой в файле Xml
            table.WriteXml(path + "Employees.xml");// Записываем без схемы
            // Читаем Xml без схемы с помощью набора DataSet
            ds = new DataSet();
            ds.ReadXml(path + "Employees.xml");// Читает и без схемы
            //table.ReadXml(path + "Employees.xml");// Будет сбой при выполнении
            // Показываем прочитанную таблицу пользователю
            dataGridView.DataSource = ds.Tables[0];// Единственная в коллекции

Ситуацию с чтением DataTable.ReadXml можно исправить, если при записи сохранять в Xml сразу и схему и данные, но можно схему сохранить отдельно, а потом перед чтением данных ее загрузить.

// Путь к каталогу записи
            String path = Application.StartupPath.ToString() + @"\Data\";
            // Объект отображения
            DataGridView dataGridView = new DataGridView();
            dataGridView.Parent = this;
            dataGridView.Dock = DockStyle.Fill;
    
            // Заполняем множественный набор данными из БД
            DataSet ds = LoadDataSet();
            // Создаем объект таблицы
            DataTable table = new DataTable();
            // Копируем в таблицу данные из набора, где есть и схема 
            table = ds.Tables["Empl"].Copy();
            // Сохраняем схему в отдельном файле (сейчас или ранее!)
            table.WriteXmlSchema(path + "Schema.xml");
            // Сохраняем таблицу без схемы
            table.WriteXml(path + "Employees.xml");
    
            // Создаем новый чистый объект таблицы 
            table = new DataTable();
            // Читаем схему из файла
            table.ReadXmlSchema(path + "Schema.xml");
            // Сюда же читаем файл без схемы, теперь можно
            table.ReadXml(path + "Employees.xml");
    
            // Показываем пользователю
            dataGridView.DataSource = table;

Иногда может потребоваться записать в Xml-формате строки объектной таблицы, каждый в своем файле. Это можно сделать так

// Путь к каталогу записи
            String path = Application.StartupPath.ToString() + @"\Data\";
    
            DataSet ds = LoadDataSet();// Заполняем набор из БД
            DataSet ds1 = new DataSet();// Вспомогательный DataSet
            int i = 0;
            foreach (DataRow r in ds.Tables["Empl"].Rows)
            {
                ds1.Clear();
                DataRow[] row = { r };// Merge() ожидает массив строк
                ds1.Merge(row);// Добавляем строку из другого набора
                ds1.WriteXml(path + "row" + ++i + ".xml");// Записываем в Xml-файл
            }

Собрать Xml-строки, сохраненные предыдущим кодом, в набор данных можно так

// Путь к каталогу записи
            String path = Application.StartupPath.ToString() + @"\Data\";
            // Объект отображения
            DataGridView dataGridView = new DataGridView();
            dataGridView.Parent = this;
            dataGridView.Dock = DockStyle.Fill;
    
            // Заполняем множественный набор данными из БД
            DataSet ds = LoadDataSet();
            int count = ds.Tables["Empl"].Rows.Count;
    
            ds = new DataSet();// Новый DataSet
            for (int i = 0; i < count; i++)
            {
                DataSet ds1 = new DataSet();
                ds1.Clear();
                ds1.ReadXml(path + "row" + (i + 1) + ".xml");
                DataRow[] row = { ds1.Tables[0].Rows[0] };
                ds.Merge(row);// Добавляем в текущий DataSet
            }
            // Показываем пользователю
            dataGridView.DataSource = ds.Tables[0];
Алексей Бабушкин
Алексей Бабушкин

При выполнении в лабораторной работе упражнения №1 , а именно при выполнении нижеследующего кода:

using System;

using System.Collections.Generic;

using System.ComponentModel;

using System.Data;

using System.Drawing;

using System.Text;

using System.Windows.Forms;

using Microsoft.Xna.Framework.Graphics;

   

namespace Application1

{

    public partial class MainForm : Form

    {

        // Объявим поле графического устройства для видимости в методах

        GraphicsDevice device;

   

        public MainForm()

        {

            InitializeComponent();

   

            // Подпишемся на событие Load формы

            this.Load += new EventHandler(MainForm_Load);

   

            // Попишемся на событие FormClosed формы

            this.FormClosed += new FormClosedEventHandler(MainForm_FormClosed);

        }

   

        void MainForm_FormClosed(object sender, FormClosedEventArgs e)

        {

            //  Удаляем (освобождаем) устройство

            device.Dispose();

            // На всякий случай присваиваем ссылке на устройство значение null

            device = null;       

        }

   

        void MainForm_Load(object sender, EventArgs e)

        {

            // Создаем объект представления для настройки графического устройства

            PresentationParameters presentParams = new PresentationParameters();

            // Настраиваем объект представления через его свойства

            presentParams.IsFullScreen = false; // Включаем оконный режим

            presentParams.BackBufferCount = 1;  // Включаем задний буфер

                                                // для двойной буферизации

            // Переключение переднего и заднего буферов

            // должно осуществляться с максимальной эффективностью

            presentParams.SwapEffect = SwapEffect.Discard;

            // Устанавливаем размеры заднего буфера по клиентской области окна формы

            presentParams.BackBufferWidth = this.ClientSize.Width;

            presentParams.BackBufferHeight = this.ClientSize.Height;

   

            // Создадим графическое устройство с заданными настройками

            device = new GraphicsDevice(GraphicsAdapter.DefaultAdapter, DeviceType.Hardware,

                this.Handle, presentParams);

        }

   

        protected override void OnPaint(PaintEventArgs e)

        {

            device.Clear(Microsoft.Xna.Framework.Graphics.Color.CornflowerBlue);

   

            base.OnPaint(e);

        }

    }

}

Выбрасывается исключение:

Невозможно загрузить файл или сборку "Microsoft.Xna.Framework, Version=3.0.0.0, Culture=neutral, PublicKeyToken=6d5c3888ef60e27d" или один из зависимых от них компонентов. Не удается найти указанный файл.

Делаю все пунктуально. В чем может быть проблема?