|
При выполнении в лабораторной работе упражнения №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];