Опубликован: 05.08.2010 | Доступ: свободный | Студентов: 2010 / 47 | Оценка: 4.50 / 4.40 | Длительность: 60:26:00
Самостоятельная работа 6:

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

Рассуждения о нюансах !!!

Здесь есть один нюанс. Я не смог усмирить счетчик автозаполнения первого столбца. Если последние строки добавляешь, а потом удаляешь и сохраняешь - он все-равно не сбрасывается в значение, следующее за последним столбцом при новой выборке данных, а считает сам по себе. Как до этого счетчика добраться, может кто и догадается, я не смог.

Еще раз приведу схему связей таблиц в нашей учебной БД, из которой видно, что столбец ProductID таблицы Products является первичным ключем, но есть еще и два столбца внешних ключей

Можно не ставить автозаполнение на столбец первичного ключа, а вынести счетчик отдельно, а при выборке данных посылать SQL-команду с функцией MAX().

"SELECT MAX(ProductID)
  FROM Products"

Но для большой таблицы это не выход, хотя для ключевого столбца команда выполнится быстро. Можно вообще с ключевого столбца снять свойство только для чтения, снять автозаполнение и дать пользователю самому заполнять ключевые поля уникальными значениями. Но тогда все равно нужно проверять уникальность значений ключевого столбца, иначе будет нарушено ограничение таблицы и мы получим ряд исключений. Мы знаем, что у нас проблемные пользователи (такие же, как дороги).

Попробуйте обойти эту дырку. Одно из направлений такое

  • Добавьте в класс Form1 проекта WinForms3 примерно следующий метод, получающий через SQL-запрос максимальное значение ключевого столбца
// Максимальное значение ключа
        void MaxPrimaryKey()
        {
            OleDbConnection conn = new OleDbConnection(ConnectionString());
            OleDbCommand command = new OleDbCommand(
                "SELECT MAX(ProductID) FROM Products",
                conn);
    
            conn.Open();
            int max = (int)command.ExecuteScalar();
            conn.Close();
    
            MessageBox.Show(max.ToString());
        }
  • Добавьте в конструктор класса (или еще куда-нибудь) вызов этого метода и выполните проект

Для показанного ранее на снимке состояния БД получим такой 'непричесанный' результат


  • Далее действуйте по обстановке, может быть и другим способом

Можно, также, подсчитывать количество записей SQL-запросом

"SELECT COUNT(*) FROM
  Products"

и их считать отправной точкой для счетчика при добавлении новых записей в таблицу. Но это будет работать при условии, что ключи изначально представляют собой натуральный ряд чисел. Для нашего примера это будет уже расхождение с подсчетом максимального значения ключа


Можно попробовать что-то сделать и с таким кодом

// Все загружаем и все настраиваем
        void LoadProducts()
        {
            .........................................


            tableProducts.RowChanged += new DataRowChangeEventHandler(tableProducts_RowChanged);
        }
    
        void tableProducts_RowChanged(object sender, DataRowChangeEventArgs e)
        {
            if (e.Row["ProductName"].ToString() == String.Empty)
                e.Row.Delete();
            // А что дальше???
        }
  • Думайте!!! Классики хотели, я и сподобился...
Алексей Бабушкин
Алексей Бабушкин

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

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