静态库在链接阶段,会将汇编生成的目标文件.o与引用到的库一起链接打包到可执行文件中。因此对应的链接方式称为静态链接。
首先在头文件下添加一个头文件static.h
头文件static.h的内容如下:
int sum(int a, int b);//声明函数
在源文件下添加一个源文件static.cpp,内容如下:
#include "static.h"
#include "stdafx.h"
int sum(int x, int y){
return x + y;}
在菜单栏选择“生成”->“生成解决方案”就可以了。
然后打开工程文件夹,在Debug目录下(编译选择的是默认的Debug和Win32)就可以看到一个和项目名称相同的lib文件:
这样供给别人调用的lib文件就生成好了,下面讲如何调用这个静态库文件。
首先新建另外的项目TestCallLib1,这个项目将使用我们刚刚生成的静态库
将头文件static.h和静态库TestLib1.lib拷到TestCallLib1\TestCallLib1目录下
首先将主函数写好
有两种方法调用静态库:
转存失败重新上传取消
转存失败重新上传取消
转存失败重新上传取消
运行得到结果!
创建动态库关键是导出函数,DLL中导出函数的声明有两种方式:
需要自己手动添加四个文件
头文件dllgenerator.h,内容:
int Add(int, int);int Mul(int, int);
函数定义文件dllgenerator.cpp,内容:
int Add(int a, int b){return a + b;}
int Mul(int c, int d){return c * d;}
dllmain.cpp : 定义 DLL 应用程序的入口点
// dllmain.cpp : 定义 DLL 应用程序的入口点。
#include
BOOL APIENTRY DllMain(HMODULE hModule,
DWORD ul_reason_for_call,
LPVOID lpReserved
)
{
switch (ul_reason_for_call)
{
case DLL_PROCESS_ATTACH:
case DLL_THREAD_ATTACH:
case DLL_THREAD_DETACH:
case DLL_PROCESS_DETACH:
break;
}
return TRUE;
}
需包含库
在此文件第一行引号中填你的项目名称即可,EXPORTS下面列出要生成的函数名称 格式:“名称 @序号”)
LIBRARY "TestDll1"
EXPORTS
Add @1
Mul @2
点击菜单栏 生成 -> 生成解决方案,我们的动态库就生成成功了
新建项目过程同上!这里的关键是头文件中需要加入关键字__declspec(dllexport),这个关键字是导出函数的关键!
新建项目DLLTestor,将刚刚生成的.dll、.lib、.h文件拷贝到DLLTestor\DLLTestor文件下( .lib、.h 文件不是必须拷入的,我这里并没有拷入)
工程调用dll时首先在工程文件目录中查找dll,找不到后在C:\Windows\System32 中找。所以我们自己项目简单调用dll时就把生成的dll文件复制到工程目录中,如果经常用可以把dll文件放到C:\Windows\System32中
调用动态库是调用 .dll、 .lib、 .h 三个文件
编好代码并设置.lib和.h文件的调用路径
右键“目录”“属性”选择“链接器”->“输入”,在“附加依赖项”这里添加要调用的lib文件的名字:TestDll1.lib
运行得到结果转存失败重新上传取消
转存失败重新上传取消
转存失败重新上传取消
动态加载是指在生成可执行文件时不将所有程序用到的函数链接到一个文件,因为有许多函数在操作系统带的dll文件中,当程序运行时直接从操作系统中找。
而静态加载就是把所有用到的函数全部链接到exe文件中。动态加载是只建立一个引用的接口,而真正的代码和数据存放在另外的可执行模块中,在运行时再装入;
而静态加载是把所有的代码和数据都复制到本模块中,运行时就不再需要库了。
#include "stdafx.h"
#include "dllgenerator.h"
#include "windows.h"
int _tmain(int argc, CHAR* argv[]){
printf("Hello World!\n");
HMODULE hmod = LoadLibrary("TestDll1.dll"); // //用于加载dll
typedef int(*LoadProc)(int x, int y);
LoadProc Load_proc = (LoadProc)GetProcAddress(hmod, "Add");
//GetProcAddress()用于获得函数地址
int iRet = Load_proc(3, 5); //得到地址后调用该函数,返回较大值
printf("the Add the value is:%x\n", iRet);
return 0;
}
需要理解调用动态库主要用到三个函数,加载LoadLibrary、调用GetProcAddress、释放FreeLibrary!
别忘了在右键“目录”“属性”选择“链接器”->“输入”,在“附加依赖项”这里添加要调用的lib文件的名字:TestDll1.lib
大家在制作过程中可能会遇到如下的问题
把字符集从unicode改成多字节字符集。
目前以lib后缀的库有两种,一种为静态链接库(Static Libary,以下简称“静态库”),另一种为动态连接库(DLL,以下简称“动态库”)的导入库(Import Libary,以下简称“导入库”)。静态库是一个或者多个obj文件的打包,所以有人干脆把从obj文件生成lib的过程称为Archive,即合并到一起。比如你链接一个静态库,如果其中有错,它会准确的找到是哪个obj有错,即静态lib只是壳子。动态库一般会有对应的导入库,方便程序静态载入动态链接库,否则你可能就需要自己LoadLibary调入DLL文件,然后再手工GetProcAddress获得对应函数了。有了导入库,你只需要链接导入库后按照头文件函数接口的声明调用函数就可以了。导入库和静态库的区别很大,他们实质是不一样的东西。静态库本身就包含了实际执行代码、符号表等等,而对于导入库而言,其实际的执行代码位于动态库中,导入库只包含了地址符号表等,确保程序找到对应函数的一些基本地址信息。
一般的动态库程序有lib文件和dll文件。lib文件是必须在编译期就连接到应用程序中的,而dll文件是运行期才会被调用的。如果有dll文件,那么对应的lib文件一般是一些索引信息,具体的实现在dll文件中。如果只有lib文件,那么这个lib文件是静态编译出来的,索引和实现都在其中。
转自:https://www.cnblogs.com/zqh20145320/p/6772985.html
仅供学习交流,如有侵权,请联系删除。