При выполнении в лабораторной работе упражнения №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" или один из зависимых от них компонентов. Не удается найти указанный файл. Делаю все пунктуально. В чем может быть проблема? |
Редактирование данных 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];