Приветствую Вас Гость | RSS

Меню сайта

Реклама

Категории раздела
ADO [17]
ASCII и CSV [12]
Access [20]
Alias [24]
BDE [37]
BLOB поля [19]
Clipper [2]
DB2 [2]
DBASE и DBF [26]
Fox Pro [1]
Interbase [21]
MSSQL [0]
ODBC [10]
Oracle [0]
Paradox [0]
SQL [29]
Sybase [1]
База данных [0]
Закладки [2]
Записи [0]
Индексы [10]
Компоненты и Базы данных [0]
Модуль данных [3]
Отчеты [2]
Ошибки БД [17]
Поиск [16]
Поля [0]
Сортировка и Фильтр [6]
Таблицы [0]

Наш опрос
Какие компоненты добавлять больше?
Всего ответов: 48

Статистика

Онлайн всего: 1
Гостей: 1
Пользователей: 0

Форма входа

Главная » Статьи » Базы данных » BDE

Обратные вызовы BDE32 для получения статуса операций
Данный совет показывает как в Delphi 2.01 можно использовать функцию BDE DbiCallBack для получения значения линейки прогресса при длительных пакетных операциях, связанных с движением данных. 

Дополнительная документация, описывающая вызовы функций BDE, находится в файле BDE32.HLP (расположенном в каталоге, где установлен 32-битный IDAPI). 

При создании функций обратного вызова BDE, BDE будет осуществлять "обратный вызов" функций вашего приложения, позволяя тем самым извещать ваше приложение о происходящих событиях, а в некоторых случаях передавать информацию обратно BDE. 

BDE определяет несколько возвращаемых типов, которые могут быть установлены для обратного вызова:

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

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

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

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

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

----- Демонстрационный код ---------



unit Testbc1;

interface

uses
  Windows, Messages, SysUtils, Classes, Graphics, Controls,
  Forms, Dialogs, StdCtrls, Grids, DBGrids, DB, DBTables, ComCtrls;

type
  TForm1 = class(TForm)
  Table1: TTable;
  BatchMove1: TBatchMove;
  Table2: TTable;
  Button1: TButton;
  ProgressBar1: TProgressBar;
  procedure Button1Click(Sender: TObject);
  private
  { Private declarations }
  public
  { Public declarations }
  end;

var
  Form1: TForm1;

implementation

uses Bde; {Здесь расположены Dbi Types и Procs}

{$R *.DFM}

{тип структуры данных для сохранения информации о предыдущем обратном вызове}
type
  TDbiCbInfo = record
  ecbType: CBType;
  iClientData: longint;
  DataBuffLn: word;
  DataBuff: pCBPROGRESSDesc;
  DbiCbFn: pointer;
  end;
type
  PDbiCbInfo = ^TDbiCbInfo;

  {Наша функция обратного вызова}

function DbiCbFn(ecbType: CBType;
  iClientData: Longint;
  CbInfo: pointer): CBRType stdcall;
var
  s: string;
begin
  {Проверяем, является ли тип обратного вызова тем, который мы ожидаем}
  if ecbType = cbGENPROGRESS then
  begin
  {если iPercentDone меньше нуля, извлекаем число}
  {обработанных записей из параметра szMsg}
  if pCBPROGRESSDesc(cbInfo).iPercentDone < 0 then
  begin
  s := pCBPROGRESSDesc(cbInfo).szMsg;
  Delete(s, 1, Pos(': ', s) + 1);
  {Вычислям процент выполненного и изменяем линейку прогресса}
  Form1.ProgressBar1.Position :=
  Round((StrToInt(s) / Form1.Table1.RecordCount) * 100);
  end
  else
  begin
  {Устанавливаем линейку прогресса}
  Form1.ProgressBar1.Position :=
  pCBPROGRESSDesc(cbInfo).iPercentDone;
  end;
  end;
  {существовал ли предыдущий зарегистрированный обратный вызов?}
  {если так - осуществляем вызов и возвращаемся}
  if PDbiCbInfo(iClientData)^.DbiCbFn <> nil then
  DbiCbFn :=
  pfDBICallBack(PDbiCbInfo(iClientData)^.DbiCbFn)
  (ecbType,
  PDbiCbInfo(iClientData)^.iClientData,
  cbInfo)
  else
  DbiCbFn := cbrCONTINUE;
end;

procedure TForm1.Button1Click(Sender: TObject);
var
  CbDataBuff: CBPROGRESSDesc; {Структура DBi}
  {структура данных должна хранить информацию о предыдущем обратном вызове}
  OldDbiCbInfo: TDbiCbInfo;
begin
  {Убедимся в том, что перемещаемая таблица открыта}
  Table1.Open;
  {Убедимся в том, что таблица-приемник закрыта}
  Table2.Close;
  {получаем информацию о любом установленном обратном вызове}
  DbiGetCallBack(Table2.Handle,
  cbGENPROGRESS,
  @OldDbiCbInfo.iClientData,
  @OldDbiCbInfo.DataBuffLn,
  @OldDbiCbInfo.DataBuff,
  pfDBICallBack(OldDbiCbInfo.DbiCbFn));
  {регистрируем наш обратный вызов}
  DbiRegisterCallBack(Table2.Handle,
  cbGENPROGRESS,
  longint(@OldDbiCbInfo),
  SizeOf(cbDataBuff),
  @cbDataBuff,
  @DbiCbFn);

  Form1.ProgressBar1.Position := 0;
  BatchMove1.Execute;

  {если предыдущий обратный вызов существовал - вновь устанавливаем его,}
  {в противном случае "отрегистрируем" наш обратный вызов}
  if OldDbiCbInfo.DbiCbFn <> nil then
  DbiRegisterCallBack(Table2.Handle,
  cbGENPROGRESS,
  OldDbiCbInfo.iClientData,
  OldDbiCbInfo.DataBuffLn,
  OldDbiCbInfo.DataBuff,
  OldDbiCbInfo.DbiCbFn)
  else
  DbiRegisterCallBack(Table2.Handle,
  cbGENPROGRESS,
  longint(@OldDbiCbInfo),
  SizeOf(cbDataBuff),
  @cbDataBuff,
  nil);

  {Показываем наш успех!}
  Table2.Open;

end;

end.
Категория: BDE | Добавил: Skinner (05.07.2008)
Просмотров: 367 | Рейтинг: 0.0/0
  Delphi Lab   Главная   Регистрация   Вход  
Интересная Цитата

Поиск

Магазин


Copyright MyCorp © 2024 Хостинг от uCoz