Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Методичка Lab1_ DLL (для ВМ-81,82).doc
Скачиваний:
26
Добавлен:
12.04.2015
Размер:
194.56 Кб
Скачать

Создание библиотеки dll

  • в Delphi:

1. Закройте все проекты в Delphi.

2. Выберите пункт меню File/ New/ Other… Щелкните на значкеDLL Wizard.

3. В открывшемся окне библиотеки наберите:

3.1) В первой строке замените стандартное имя библиотеки Project1на новое имя.

3.2) Перед словами beginendнаберите ваши подпрограммы.

3.3.) Перед словами beginendнаберите директиву экспорта

exports список_имен_эксопртируемых_подпрограмм;

чтобы имена подпрограмм были «видимы» за пределами библиотеки.

Например, текст библиотеки будет выглядеть так (цветом выделено то, что добавилось):

library MyDLL;

uses

SysUtils, Classes, Forms, Windows;

var если нужны переменные – описываются здесь;

procedure MyProc(var x,y,z: Integer); stdcall;

begin

z := x+y;

end;

exports

MyProc;

begin

end.

4. Сохраните проект библиотеки под именем, которое вы записали в п.3.1, и скомпилируйте (запускать по F9 смысла нет, так какDLL– это не исполнимый файл). В результате должен появиться файл с расширениемDLL.

  • в C++ Builder:

1. Закройте все проекты.

2. Выберите пункт меню File/ New/ Other… Щелкните на значкеDLL Wizard.

3. В открывшемся окне библиотеки после всех стандартных строк наберите свои функции

extern “C” __declspec(dllexport) ИмяФункции(параметры)

{

тело функции

};

Директива extern “С” __declspec(dllexport)нужна,чтобы имена функций были доступны из библиотеки.

Если extern “С”не указывать, то компилятор С++ сгенерирует так называемоерасширенное имяфункции (decoratedname), которое включает имя класса, имя функции, типы параметров. Расширенное имя нельзя использовать в других языках, кроме С++. Чтобы библиотекуDLLмогли использовать программы, написанные на других языках программирования, нужно запретить искажение имен. Для этого указывается модификаторextern “С”.

4. Сохраните проект библиотеки, и скомпилируйте (запускать по F9 смысла нет, так какDLL– это не исполнимый файл). В результате должен появиться два файла, которые будут нужны для подключения к проектам: с расширением .DLLи .LIB.

  • в Visual C++:

В VisualC++ можно создавать два типа динамических библиотек: обычныеDLLиDLL-расширения.DLL-расширения способны экспортировать С++ классы, функции-члены и переопределенные функции. ОбычнаяDLLможет экспортировать только обычные функции и глобальные переменные. Внутри обычнойDLLможно использовать классы С++ иMFC, но не экспортировать их. В лабораторной работе мы будем делать обычнуюDLL.

Объявления функций делаются так же, как в C++Builder.

Неявное связывание dll

  • в Delphi:

1. Создайте приложение (оконное или консольное), к которому будем подключать DLL.

2. Сохраните проект.

3. В папку с проектом скопируйте файл динамической библиотеки DLL.

4. В тексте программы, в секции описания процедур и функций, наберите заголовки подпрограмм из библиотеки DLL. Вместо тела подпрограммы записывается директива импортаexternal ‘ИмяБиблитоеки.dll’. Например,

procedure MyProc(var x,y,z: Integer); stdcall; external ‘MyDll.dll’;

И так для каждой подпрограммы, импортируемой из библиотеки.

5. Теперь в тексте можно вызывать процедуры и функции из библиотеки.

6. Сохраните и скомпилируйте проект.

Для отладки DLL:

1. Откройте проект динамической библиотеки.

2. В меню выберите пункт Run/Parameters/ вкладкуLocal. В параметреHost Applicationукажите путь кEXE-шнику, который подключает вашу библиотеку.

3. В тестируемой подпрограмме установите точку останова.

4. Запустите библиотеку на выполнение (F9). Будет запущенEXE-шник, и в момент вызова вашей процедуры или функции сработает точка останова.

Если вы отлаживаете ассемблерные коды, то для показа окна отладчика вызовите из главного меню View/ Debug Windows / CPU, и для сопроцессора View/ Debug Windows / FPU.

  • в C++ Builder:

1. Создайте приложение (оконное или консольное), к которому будем подключать DLL.

2. Сохраните проект.

3. В папку с проектом скопируйте файлы DLLиLIBиз папки с библиотекой.

4. Подключите библиотечный файл LIBк проекту. Это можно сделать в окнеProjectManager, выбрав в контекстном меню командуAdd, или в файл проекта вписать строку

USELIB("имя_библиотеки.dll")

5. В текст программы добавьте заголовки импортируемых из библиотеки DLLфункций.

extern “C” __declspec(dllimport) ИмяФункции(параметры);

И так для каждой подпрограммы, импортируемой из библиотеки.

5. Теперь в тексте можно вызывать процедуры и функции из DLL.

6. Сохраните и скомпилируйте проект.

Отладка DLL выполняется так же, как вDelphi.

Замечание: бывает удобно все объявления экспортируемых изDLLфункций объединить в единый заголовочный файлh, и подключать его (меняя, где нужно,dllexportнаdllimport) во все проекты, вызывающие этуDLLи в самуDLL. Замена словdlexportнаdllimportпри объявлении функций реализуется за счет макроопределений. Пример заголовочного файлаh:

#ifdef __DLL__

#define DLL_SPEC extern "C" __declspec(dllexport)

#else

#define DLL_SPEC extern "C" __declspec(dllimport)

#endif

// далее определяются переменные и функции, например

DLL_SPEC int MyVariable;

DLL_SPEC int MyFunc(int x);

Если определен идентификатор __DLL__, макроопределениеDLL_SPECраскрывается черезdllexport, в противном случае – черезdllimport. Таким образом, при включении заголовочного файла в проектDLL, нужно определить идентификатор__DLL__, а при включении заголовочного файла в обычную программу – не нужно. То есть, первой строкойDLLдолжна идти

#define __DLL__

и еще директива #include “имя_заголовочного_файла.h”.

Для явного связыванияв текст клиентской программыLIB-файл и объявления импортируемых функций включать не нужно. Подключение и работа сDLLвыполняется через вызовыLoadLibrary,GetProcAddressиFreeLibrary. ФункцияLoadLibraryпытается отыскать и загрузить динамическую библиотеку. Ей передается строка, содержащая путь и имя файлаDLL. Функция возвращает описатель экземпляра типаHINSTANCE, который используется в вызовахGetProcAddress. После окончания работы программы следует отключиться отDLL, вызвав функциюFreeLibrary.

Пример: допустим, чтоDLLэкспортирует функцию так:

extern "C" __declspec(dllexport) double SquareRoot(double x);

Упрощённый пример, без проверок на ошибки, взят из [3]:

//объявление указателя на функцию

void (__stdcall *MyFunction)(HWIND);

//загрузка DLL

HINSTANCE dllInstance=LoadLibrary("mylib.dll");

//получение адреса функции

MyFunction=(void(__stdcall*)(HWIND)GetProcAddress(dllInstance,"MyFunc");

//вызов функции

MyFunction(Application->Handle);

//выгрузка DLL

FreeLibrary(dllInstance);

В VisualC++ можно использовать проверки корректности выполнения, например:

typedef double (SQRTPROC)(double);

HINSTANCE hInstance;

SQRTPROC* pFunction;

VERIFY( hInstance=::LoadLibrary("c:\\winnt\\system\\mylib.dll"));

VERIFY( pFunction=(SQRTPROC*)::GetProcAddress((HMODULE)hInstance,"SquareRoot"));

double d=(*pFunction)(0.225); // вызов функции из DLL

Приложение 2