В данной статье мы обратим внимание на компонент TADOCommand и
использование языка SQL DDL (Data Definition Language), с целью
помочь Вам с проблемой переноса данных BDE/Paradox в ADO/Access.
Язык определения данных (Data Definition Language)
Не многие программисты создают базу данных программным путём,
большинство из нас для этого используют некую визуальную среду
наподобие MS Access для построения файла MDB. Но иногда нам всё таки
приходится создавать и удалять базу данных, а так же объекты базы
данных программным путём. Для этого используется наиболее
распространённая на сегодняшний день технология Structured Query
Language Data Definition Language (SQL DDL). Выраджения языка
определения данных (DDL) - это SQL выражения, которые поддерживают
определения или объявления объектов базы данных (например, CREATE
TABLE, DROP TABLE, CREATE INDEX либо подобные им).
В рамки данной статьи не входит детальное ознакомление с языком
DDL. Если Вы знакомы с языком SQL DML (Data Manipulation Language -
это выражения типа SELECT, UPDATE и DELETE), то DDL не будет для Вас
серьёзным барьером. Обратите внимание, что работа с DDL может быть
весьма ухищрённой, так как каждый производитель базы данных може
включать в неё собственные расширения для SQL.
Давайте взглянем на простейший пример выражения CREATE TABLE:
CREATE TABLE PhoneBook(
Name TEXT(50)
Tel TEXT(50));
|
Данное DDL выражение (для MS Access) в время выполнения создаст
новую таблицу с названием PhoneBook. Таблица PhoneBook будет иметь
два поля: Name и Tel. Оба поля имеют строковый тип (TEXT) и размер
поля 50 символов.
TFieldDef.DataType
Очевидно, что в Access тип данных, представленный строкой это
TEXT. В Paradox это STRING. Чтобы передать таблицы Paradox в Access,
нам необходимо знать какие типы данных присутствуют и,
соответственно их имена. При работе в BDE с таблицами Paradox,
TFieldDef.DataType определяет тип физического поля в (dataset)
таблице. Поэтому для успешного перенесения данных из таблиц Paradox
в Access Вам необходимо создать функцию, которая бы преобразовывала
соотвествующие типы полей Paradox в типы Access.
Давайте посмотрим на пример функции, которая проверяет тип поля
(fd) и возвращает соответствующий тип Access, а заоодно и размер
поля, который необходим для выражения CREATE TABLE DDL.
function AccessType(fd:TFieldDef):string;
begin
case fd.DataType of
ftString: Result:='TEXT('+IntToStr(fd.Size)+')';
ftSmallint: Result:='SMALLINT';
ftInteger: Result:='INTEGER';
ftWord: Result:='WORD';
ftBoolean: Result:='YESNO';
ftFloat : Result:='FLOAT';
...
else
Result:='TEXT(50)';
end;
end;
|
ADOX
ADOX - это расширения ADO для Data Definition Language а так же
для модели защиты (ADOX). ADOX предоставляет разработчикам богатый
набор инструментов для получения доступа к структуре, модели защиты,
а так же процедурам, хранимым в базе данных.
Для использования ADOX в Delphi, Вам необходимы установить
библиотеку типа ADOX.
- Select Project | Import Type Library
- Выберите "Microsoft ADO Ext 2.x for DDL and Security (Version
2.x)"
- Измените "TTable" на "TADOXTable"
- Измените "TColumn" на "TADOXColumn"
- Измените "TIndex" на "TADOXIndex"
- Нажмите кнопку Install (перекомпиляция пакетов (packages))
- Нажмите один раз OK и дважды Yes
- File | Close All | Yes
На вершине объектной модели ADOX находится объект Catalog. Он
обеспечивает доступ к набору Таблиц (Tables), Видов (Views) и
Процедур, который используется для работы со структурой базы данных,
а так же к набору Пользователей (Users) и рупп (Groups), которые
используются для авторизации доступа. Каждый объект Catalog связан
только с одним подключением к источнику данных.
Давайте оставим ADOX (пока) и перейдём к ADOExpress.
TADOCommand
В ADOExpress компонент TADOCommand - это VCL представление
объекта ADO Command. Объект Command представляет команду (запрос или
выражение), которая может быть обработана источником данных. Команды
могут быть выполнены методом Execute, используемым в ADOCommand.
TADOCommand чаще всего используется для исполнения команд языка
определения данных (DDL) SQL. Свойство CommandText содержит в себе
саму команду. Свойство CommandType используется для того, как
интерпретировать свойство CommandText. Тип cmdText используется для
указания инструкции DDL. Впринципе, использовать компонент
ADOCommand для получения данных из таблицы, запросов или хранимых
процедур не имеет смысла, но никто не запрещает Вам пользоваться
данным компонентов и в таких целях.
Итак, самое время приступить к реальному программированию...
Приведённый ниже проект демонстрирует следующее:
Получение списка всех таблиц из BDE, использование TFieldDefs
чтобы получить определения (имя, тип данных, размер, и т.д.) полей в
таблице, создание инструкции CREATE TABLE и копирование данных из
таблицы BDE/Paradox в таблицу ADO/Access.
Давайте решим эту задачу по шагам:
GUI
Запускаем Delphi - получаем новый проект с пустой формой.
Добавляем две кнопки, один ComboBox и один компонент Memo. Далее
добавляем компоненты TTable, TADOTable, TADOConnection и
TADOCommand. Чтобы установить следующие свойства, используем Object
Inspector (оставьте все другие свойства как есть - например, Memo
будет иметь имя по умолчанию: Memo1):
Button1.Caption = 'Construct Create command'
Button2.Caption = 'Create Table and copy data'
ComboBox.name = cboBDETblNames;
ADOConnection1.ConnectionString = ...
TADOTable.name = ADOTable
ADOTable.Connection = ADOConnection1
TADOCommand.name = ADOCommand
ADOCommand.Connection = ADOConnection1
TTable.name = BDETable
BDETable.DatabaseName = 'DBDEMOS'
|
Пример:
Для получения списка таблиц, связанных с данной базо данных
(DBDEMOS) мы воспользуемся следующим кодом (OnCreate для формы):
procedure TForm1.FormCreate(Sender: TObject);
begin
Session.GetTableNames('DBDEMOS', '*.db', False,
False, cboBDETblNames.Items);
end;
|
В самом начале ComboBox содержит имена таблиц (Paradox) в базе
данных DBDEMOS. В нижеприведённом коде мы выберем таблицу Country.
Следующая наша задача - это создание инструкции CREATE TABLE DDL.
Это делается в процедуре OnClick кнопки 'Construct Create command':
//Кнопка 'Construct Create command'
procedure TForm1.Button1Click(Sender: TObject);
var
i: integer;
s: string;
begin
BDETable.TableName:=cboBDETblNames.Text;
BDETable.FieldDefs.Update;
s:='CREATE TABLE ' + BDETable.TableName + ' (';
with BDETable.FieldDefs do
begin
for i:=0 to Count-1 do
begin
s:=s + ' ' + Items[i].name;
s:=s + ' ' + AccessType(Items[i]);
s:=s + ',';
end;
s[Length(s)]:=')';
end;
Memo1.Clear;
Memo1.lines.Add (s);
end;
|
Вышеприведённый код просто анализирует определения полей для
выбранной таблицы (cboBDETblNames) и генерирует строку, которая
будет использоваться свойством CommandText компоненты TADOCommand.
Например, когда Вы выбираете таблицу Country, то Memo будет
заполнен следующей строкой:
CREATE TABLE country(
Name TEXT(24),
Capital TEXT(24),
Continent TEXT(24),
Area FLOAT,
Population FLOAT)
|
И в заключении, пример для кнопки 'Create Table and copy data' ,
которая удаляет таблицу (DROP..EXECUTE), создаёт таблицу
(CREATE..EXECUTE), и затем копирует данные в новую таблицу
(INSERT...POST). Так же присутствует некоторая обработка ошибок, но
код будет выходить на ошибку, если, например, (новая) таблица ещё не
существует (в случае удаления).
//Кнопка 'Create Table and copy data'
procedure TForm1.Button2Click(Sender: TObject);
var
i: integer;
tblName: string;
begin
tblName:=cboBDETblNames.Text;
//обновляем
Button1Click(Sender);
//удаление & создание таблицы
ADOCommand.CommandText:='DROP TABLE ' + tblName;
ADOCommand.Execute;
ADOCommand.CommandText:=Memo1.Text;
ADOCommand.Execute;
ADOTable.TableName:=tblName;
//копируем данные
BDETable.Open;
ADOTable.Open;
try
while not BDETable.Eof do
begin
ADOTable.Insert;
for i:=0 to BDETable.Fields.Count-1 do
begin
ADOTable.FieldByName
(BDETable.FieldDefs[i].name).Value :=
BDETable.Fields[i].Value;
end;
ADOTable.Post;
BDETable.Next
end;
finally
BDETable.Close;
ADOTable.Close;
end;
end;
|
Вот и всё. Теперь проверьте Вашу базу данных Access...вуаля :)
теперь в ней находится таблица Country со всеми данными из DBDEMOS.
Однако некоторые вопросы остались без ответа, например: как
добавлять индексы в таблицу (CREATE INDEX ON ...), или как создавать
пустую базу данных Access.
|