本文使用的是dev-c++,如果涉及到VC++中不一样的操作,也会适当进行区分。
项目一:创建DLL
1、创建一个DLL类型的项目,当前命名为dlltest,并选择合适的路径进行保存。
2、在生成的预设置代码中,加入如下代码
//这是头文件dll.h
#ifndef _DLL_H_
#define _DLL_H_
#if BUILDING_DLL
#define DLLIMPORT __declspec(dllexport)
#else
#define DLLIMPORT __declspec(dllimport)
#endif
class DLLIMPORT DllClass
{
public:
DllClass();
virtual ~DllClass();
void HelloWorld(char* info);
};
extern "C"
{
DLLIMPORT int HW(int n);
}
DLLIMPORT int func(int n);
#endif
/*这是主体文件dllmain.cpp */
#include "dll.h"
#include <windows.h>
DllClass::DllClass()
{
}
DllClass::~DllClass()
{
}
void DllClass::HelloWorld(char* info)
{
MessageBox(0, info,"Hi",MB_ICONINFORMATION);
}
DLLIMPORTint HW(int n)
{
return n;
}
DLLIMPORT int func(int n)
{
returnn; }
BOOL WINAPI DllMain(HINSTANCE hinstDLL,DWORD fdwReason,LPVOID lpvReserved)
{
switch(fdwReason)
{
case DLL_PROCESS_ATTACH:
{
break;
}
case DLL_PROCESS_DETACH:
{
break;
}
case DLL_THREAD_ATTACH:
{
break;
}
case DLL_THREAD_DETACH:
{
break;
}
}
/* Return TRUE on success, FALSE on failure */
return TRUE;
}
在上面的代码中,我们加入了HW和func两个导出函数,以及一个DllClass(自动生成)导出类。
点击编译后,我们可以在项目文件夹中,看到dlltest.dll,这就是我们需要的目标动态链接库。libdlltest.a则是vc里需要用到的lib文件。
3、extern "C"说明
当前可以用记事本打到libdlltest.def文件,可以看到如下内容:
加了extern "C"的HW函数地址偏移量还是HW,没有加extern "C"的func函数,地址偏移量变成了_Z4funci。这个地址在动态调用导出函数的过程中会用到。
项目二:动态调用dll导出的函数
1、再创建一个C++项目,将项目一生成的dll文件放入项目文件夹中:
2、使用LoadLibrary和和GetProcAddress动态载入动态链接库,并调用导出的函数:
#include <iostream>
#include <windows.h>
using namespace std;
int main()
{
HMODULE hMod=LoadLibrary("dlltest.dll");
if(hMod==NULL)
{
cerr<<"load lib error";
return 1;
}
Func f=(Func)GetProcAddress(hMod,"HW");
cout<<f(200);
FreeLibrary(hMod);
return 0;
}
在GetProcAddress中,调用HW函数可以直接传入偏移量HW;
如果调用func函数,则要传入偏移量"_Z4funci";因为func函数没有声明为extern "C"。
3、特别备注,当前这种方式无法使用LoadLibrary和GetProcAddress获出导出类。
因为GetProcAddress获取的是函数的地址偏移量,与类无关。为了可以动态使用导出的类,必须使用将一个纯虚函数做为基类,将导出创建和销毁类的函数。具体做法如下:
//dll.h
#include <stdlib.h>
#include <stdio.h>
class virtualXXX
{
public:
virtual void functionOne() = 0;
virtual void functionTwo() = 0;
};
#if defined(_WINDOWS)
#ifdef XXX_API
#define XXX_API __declspec(dllexport)
#else
#define XXX_API __declspec(dllimport)
#endif
#else
#define XXX_API
#endif
class XXX_API xxx : public virtualXXX
{
public:
void functionOne()
{
printf ( "One\n" );
}
void functionTwo()
{
printf ( "Two\n" );
}
};
extern "C" XXX_API virtualXXX * create();
extern "C" XXX_API void delete_object( virtualXXX * p );
//dll.cpp
virtualXXX * create()
{
return ( new xxx() );
}
void delete_object( virtualXXX * p )
{
if ( p )
{
delete p;
p = NULL;
}
}
动态调用:
#include <Windows.h>
typedef virtualXXX *(fun_create)(void);
fun_create* vc_create = NULL;
int main()
{
HINSTANCE dllHandle = NULL;
dllHandle = LoadLibrary( "Win32_Test_dll.dll" );
vc_create = ( fun_create* )GetProcAddress( dllHandle,"create" );
virtualXXX * xxxHandle = vc_create();
xxxHandle->functionOne();
xxxHandle->functionTwo();
delete_object(xxxHandle);
}
这个方法参考文章C++动态库导出类,及使用,博主未实际进行测试。
项目三:静态调用导出的类
静态调用dll,在VC++需要头文件、dll和对应的lib文件(即项目一中生成的 libdlltest.a)。然后再使用#pragma comment(lib,"lib文件路径")对编译器进行配置lib路径,之后再进行调用。详细过程可以参考《c++生成DLL并调用》。
本文着重调论Dev-C++下的静态调用。对于MinGW64静态调用dll,只需要dll文件和相关的头文件,项目结构如下:
其中main.cpp中调用类的代码如下:
#include <iostream>
#include <windows.h>
#include "dll.h"
using namespace std;
int main()
{
DllClass c;
char str[]="hello";
c.HelloWorld(str);
return 0;
}
可以看到,无需在代码中进行任何设置。因为只有在链接的过程c++才会去找DllClass这个类的真实地址。
在编译成exe时,有如下两种方法:
方法一、可以使用命令行进行编译:
通过cmd进入main.cpp所在文件夹路径,运行:g++ -o main.exe main.cpp -I . -L . -ldlltest
即可编译生成可执行文件exe。编译参数说明如下:
-I搜索头文件的目录
-I .在当前文件夹下搜索头文件
-L搜索动态库的目录
-L .在当前文件夹下搜索动态库
方法二、将参数加入编译选项中:
如果觉得用命令行编译太麻烦,可以将-I -L和-l加入Dev-C++的编译器选项中。
这样点击"编译运行"就可以正确找到对应的dll进行编译链接,正确生成exe文件。
本文关于Dev-C++创建并调用动态链接库dll到这里就结束了,欢迎大家指正:)