一、概念
1.1、当我们建立一个项目(应用程序)它可以有很多种生成类型(应用程序.exe、动态库.dll、静态库lib),可以在创建向导时指定也可以在配置属性里更改。以vs2010为例,右键点击项目属性->配置属性->右侧配置类型中选择你需要生成的类型。
1.2、当项目生成类型是静态库(.lib),程序会生成一个以项目名命名后缀为.lib的静态库文件。而当项目生成类型是动态库(.dll),程序会生成两个主要文件,一个是以项目命名后缀为.lib文件(仅包含了对应DLL文件中函数的重定向位置)和一个以项目命名后缀为.dll文件。
1.3、对于静态库只有一种链接方式。而动态库,分为两种,一个是静态链接方式(静态链接),一个是动态链接方式(动态链接)。
二、实现(在一个解决方案LIBDLL下建立三个项目)
2.1、静态库链接库(项目名StaticLib)
addS.h
#pragma once
#ifndef STATICLIB_ADD_H_
#define STATICLIB_ADD_H_
extern "C" int addS(int x,int y);
#endif
addS.cpp
#include "addS.h"
int addS(int x,int y)
{
return x+y;
}
2.2、动态链接库(项目名DLL)
addD.h
#pragma once
#ifndef DLL_ADD_H_
#define DLL_ADD_H_
extern "C" int __declspec(dllexport) addD(int x,int y);
#endif
addD.cpp
#include "addD.h"
int addD(int x,int y)
{
return x+y;
}
2.3、应用程序(项目名LIBDLL)
main.cpp
#include<iostream>
//静态库的链接方法
//可将路径配置到:"C/C++->常规->附加包含目录"中,则可直接写成#include "addS.h"
#include "../StaticLib/addS.h"
//可将lib库配置到:"链接器->输入->附加依赖项"中,则下面这句就可省略
#pragma comment(lib,"../x64/Debug/StaticLib.lib")
//动态库的静态链接方法
extern "C" int __declspec(dllimport) addD(int ,int );
#pragma comment(lib,"../x64/Debug/DLL.lib")//亦可配置到附加依赖项中
//动态库的动态链接方法
#include <Windows.h>
typedef int(*lpAddDFun)(int,int);
using namespace std;
int main(int argc,char** argv)
{
//静态库的链接方法
cout<<addS(1,7)<<endl;
//动态库的静态链接方法
cout<<addD(1,7)<<endl;
//动态库的动态链接方法
HINSTANCE hDll;
lpAddDFun addDFun;
hDll = LoadLibrary(L"../x64/Debug/DLL.dll");
if (hDll != NULL)
{
addDFun = (lpAddDFun)GetProcAddress(hDll,"addD");
if (addDFun != NULL)
cout<<addDFun(1,7)<<endl;
FreeLibrary(hDll);
}
return 0;
}
三、分析
我们常用的方式是静态库链接和动态库的静态链接方式。观察可发现这两种方式调用实现非常相似。但是静态链接在头文件声明是以下形式
extern "C" int __declspec(dllexport) addD(int x,int y);
而在调用中的声明却是以下形式
extern "C" int __declspec(dllimport) addD(int x,int y);
为了方便用户调用我们可以简单改写DLL项目
我们新增一个StdAfx.h文件
StdAfx.h
#pragma once
#ifndef DLL_STDAFX_H_
#define DLL_STDAFX_H_
#ifndef DLL_ADD_H_ADDD_
#define DLL_ADD_H_ADDD_
#endif
#endif
addD.h
#pragma once
#ifndef DLL_ADD_H_
#define DLL_ADD_H_
#ifdef DLL_ADD_H_ADDD_
#define DLL_ADD_H_ADDD_EI_ __declspec(dllexport)
#else
#define DLL_ADD_H_ADDD_EI_ __declspec(dllimport)
#endif
extern "C" int DLL_ADD_H_ADDD_EI_ addD(int x,int y);
#endif
addD.cpp
#include "StdAfx.h"
#include "addD.h"
int addD(int x,int y)
{
return x+y;
}
其实就是用了个简单的宏定义技巧,在DLL项目中我们定义了宏 DLL_ADD_H_ADDD_,则DLL_ADD_H_ADDD_EI_被翻译成__declspec(dllexport)。而在LIBDLL项目中,我们没有定义宏 DLL_ADD_H_ADDD_,则则DLL_ADD_H_ADDD_EI_被翻译成__declspec(dllimport)。
现在我们调用就可以直接包含头文件,和静态库的链接的调用完全一样。(静态链接要把.dll文件放在.exe目录下哦)
//动态库的静态链接方法
#include "../DLL/addD.h"//亦可配置到附加包含目录中,直接写成#include "addD.h"
//extern "C" int __declspec(dllimport) addD(int ,int );
#pragma comment(lib,"../x64/Debug/DLL.lib")//亦可配置到附加依赖项中