KnigaRead.com/
KnigaRead.com » Компьютеры и Интернет » Программирование » Валентин Озеров - Советы по Delphi. Версия 1.4.3 от 1.1.2001

Валентин Озеров - Советы по Delphi. Версия 1.4.3 от 1.1.2001

На нашем сайте KnigaRead.com Вы можете абсолютно бесплатно читать книгу онлайн Валентин Озеров, "Советы по Delphi. Версия 1.4.3 от 1.1.2001" бесплатно, без регистрации.
Перейти на страницу:

  begin

   EncodeFieldDesc(PFieldDescList(pFieldDesc)^[I], Name,DataType, Size);

  end;

  { тип драйвера nil, т.к. поля логические }

  Check(DbiTranslateRecordStructure(nil, iFields, pFieldDesc, nil, nil, pFieldDesc));

  { здесь hCursor получает свое значение }

  Check(DbiCreateInMemTable(DBHandle, szTblName, iFields, pFieldDesc, hCursor));

 finally

  if pFieldDesc <> nil then FreeMem(pFieldDesc, iFields * SizeOf(FLDDesc));

  SetDBFlag(dbfTable, False);

 end;

end;


end.


{Данный код взят из файлов помощи Ллойда!}

Поиск 

FindKey для нескольких полей

Delphi 1 

with Table1 do

begin

 SetKey;

 FieldByName('State').AsString := 'CA';

 FieldByName('City').AsString := 'Scotts Valley';

 GotoKey;

end;

Вы не можете использовать Findkey с файлами DBase более чем для одного поля. 

oEmetb.indexName:='PrimaryKey';

if oEmeTb.findkey([prCLient,prDiv,prEme]) then

где findkey передаются параметры для Primary Keyfields.

Я обращаю ваше внимание на то, что имя индекса (Index) чувствительно к регистру, так что будьте внимательны.

Вы можете также воспользоваться oEmeTb.indexfieldnames, но убедитесь в том, что ваш список ключевых полей в точности соответствуют ключевым полям, которые вы ищете. 

oEmetb.indexfieldNames:='EmeClient;EmeDiv;EmeNo';

if oEmeTb.findkey([123,'a',96]) then 

Поиск существующей записи перед тем, как она будет вставлена

Если вы находитесь в режиме редактирования (Edit) или вставки (Insert), то при изменении режима вы автоматически делаете постинг записи. И, естественно, при наличие дубликата (неуникальности) записи, вы получите ошибку. Способ обойти это – использовать другой компонент TTable, связанный с той же таблицей, и осуществляющий по ней поиск. Этот путь самый простой и эффективный.

Воспользуйтесь двумя компонентами TTable (оба должны указывать на одну и ту же таблицу). Используйте один для поиска, а второй для редактирования.

Ваша «ключевая» таблица BDE будет автоматически генерировать исключения, если пользователь будет пытаться послать созданный им дублирующий ключ. Для установки таблицы используйте Database Desktop.

Создайте на основе поля первичный индекс (Primary Index). Затем создайте какой-то обработчик DB-исключения для нашего «нарушения уникальности».

Моя технология заключается в следующем: в отдельной форме я предлагаю пользователям ввести часть записи, которая должна быть уникальна (обычно одно поле). Затем для проверки существования я делал FindKey. Если он находился, через MessageDlg я информировал пользователя, и возвращал его на форму редактирования, не создавая новой записи. Помните, что если FindKey ничего не находит, dbCursor никуда не перемещается, и закладка не нужна. Если запись найдена, она немедленно будет отображена на форме редактирования для того, чтобы пользователь смог увидеть ее содержимое. В противном случае происходит следующее: 

Table.Append;

Table.FieldByName('KeyField').AsString := UserEntry;

{ … позволяем пользователю редактировать все остальные поля записи … }

{ в это время кнопка Cancel должна быть активной для того, чтобы дать возможность пользователю отменить ввод новой записи. }

В моей форме редактирования поле с уникальном ключем выключается (disabled) и показывается с другим цветом. Целостность соблюдена :-). 

Поиск фраз и записей переменной длины

Delphi 1 

Для текста переменной длины вы можете использовать DBmemo. Большинство людей это делают сканированием «на лету» (когда оператор постит запрос), но для реального ускорения процесса можно попробовать способ пре-сканирования, который делают «большие мальчики» (операторы больших баз данных):

1. при внесении в базу данных новой записи она сканируется на предмет определения ключевых слов (это может быть как предопределенный список ключевых слов, так и всех слов, не встречающиеся в стоп-листе [пример: «the», «of», «and"])

2. ключевые слова вносятся в список ключевых слов со ссылкой на номер записи, например, «hang»,46 или «PC»,22.

3. когда пользователь делает запрос, мы извлекаем все записи, где встречается каждое из ключевых слов, например, «hang» может возвратить номера записей 11, 46 и 22, тогда как «PC» — записи с номерами 91, 22 и 15.

4. затем мы объединяем числа из всех списков c помощью какого-либо логического оператора, например, результатом приведенного выше примера может быть запись под номером 22 (в случае логического оператора AND), или записи 11, 15, 22, 46 и 91 (в случае оператора OR). Затем извлекайте и выводите эти записи.

5. для синонимов определите таблицу синонимов (например, «hang»,"kaput»), и также производите поиск синонимов, добавляя их к тому же списку как и оригинальное слово.

6. слова, имеющие общие окончания (например, «hang» и «hanged»), можно также сделать синонимами, или, как это делает большинство систем, производить анализ окончаний слов, вычисляя корень по их перекрытию (например, слову «hang» соответствует любое слово, чьи первые 4 буквы равны «hang»).

Конечно, есть множестно технических деталей, которые необходимо учесть, например, организация списков, их эффективное управление и объединение. Оптимизация этой характеристики может вам дать очень быстрое время поиска (примером удачный реализаций могут служить двигатели поиска Nexus, Lycos или WebCrawler, обрабатывающие сотни тысяч записей в течение секунды).

dBase 

Текущий номер записи набора данных

Delphi 1

{Извлекает физический номер записи xBase. Требует наличие модулей DBITYPES, DBIPROCS, и DBIERRS в списке используемых модулей. Функция требует на входе один аргументтипа TTable (например, Table1).}

function Form1.Recno(oTable: TTable): Longint;

var

 rError: DBIResult;

 rRecProp: RECprops;

 szErrMsg: DBIMSG;

begin

 Result := 0;

 try

  oTable.UpdateCursorPos;

  rError := DbiGetRecord(oTable.Handle, dbiNOLOCK, nil, @rRecProp);

  if rError = DBIERR_NONE then Result := rRecProp.iPhyRecNum

  else case rError of

  DBIERR_BOF: Result := 1;

  DBIERR_EOF: Result := oTable.RecordCount + 1;

  else

   begin

    DbiGetErrorString(rError, szErrMsg);

    ShowMessage(StrPas(szErrMsg));

   end;

  end;

 excepton

  E: EDBEngineError do ShowMessage(E.Message);

 end;

end;

Как открыть индексированную таблицу dBase, если отсутствует файл индекса?

Nomadic советует:

Для dBase-таблицы встроенными средствами ты не перестроишь индекс, если его нет. Для этой цели мне пришлось написать процедуру для физического удаления признака индексации в самом dbf-файле и после её применения добавлять индексы заново.

Для этого в заголовок файла dbf по смещению 28(dec) записываешь 0.

По другому никак не выходит(я долго бился) — вот для Paradox таблиц все Ok.

С помощью BDE Callbacks. Пример для Delphi 2.0, на первом не проверял:

=== Callback.pas ===

unit Callback;


interface


uses BDE, Classes, Forms, DB, DBTables;


type

 TForm1 = class(TForm)

  Table1: TTable;

  procedure FormCreate(Sender: TObject);

  procedure FormDestroy(Sender: TObject);

 private

  CBack: TBDECallback; // опpеделение BDE CallBack

  CBBuf: CBInputDesc; // пpосто буфеp

  function CBFunc(CBInfo: Pointer): CBRType; // Callback-функция

 public

 end;


var

 Form1: TForm1;


implementation


{$R *.DFM}


procedure TForm1.FormCreate(Sender: TObject);

begin

 Session.Open; // В это вpемя сессия ещё не откpыта

 CBack := TBDECallback.Create(Session {Hапpимеp}, nil, cbINPUTREQ, @CBRegBuf, SizeOf(CBBuf), CBFunc, False); // Опpеделили Callback

 Table1.Open;

 //^^^^^^^^^^^ - здесь возможна ошибка с индексом, etc.

end;


procedure TForm1.FormDestroy(Sender: TObject);

begin

 CBack.Free; // Освобождаем CallBack

end;


function TForm1.CBFunc(CBInfo: Pointer): CBRType;

begin

 with PCBInputDesc(CBInfo)^ do case eCbInputId of

 cbiMDXMissing {, cbiDBTMissing - можно ещё и очищать BLOB-поля}:

Перейти на страницу:
Прокомментировать
Подтвердите что вы не робот:*