VS2010中lib与dll文件的生成与使用方法
程序员文章站
2023-12-17 12:42:10
一、 lib文件的简介
.lib是一种文件后缀,是windows操作系统的库文件,有静态lib和动态lib之分:
1)、静态lib文件:将导出的文件的声明和实...
一、 lib文件的简介
.lib是一种文件后缀,是windows操作系统的库文件,有静态lib和动态lib之分:
1)、静态lib文件:将导出的文件的声明和实现都放在lib文件中,此时lib文件主要包含函数的实现部分(cpp文件),例如类的函数定义。使用时只需配合相关的头文件,编译后程序将lib文件中的代码嵌入到宿主程序中,也就是最后的exe文件中,此时移除lib文件,程序可以正常运行,即编译后就不需要lib文件的支持了。
2)、动态lib文件:相当于是一个h头文件,用于支持相应的dll文件的运行。里面存储的是dll文件中各个导出函数的地址,达到链接主程序与dll文件的目的。
二、dll文件的生成
vs2010生成dll文件,生成dll文件的时候需要对应的lib文件才能使用,dll生成方法如下(此处只是生成部分,在使用时还需修改):
1) 新建工程,选择“win32项目”,注意不是“win32控制台项目”,下一步选择生成dll文件,其余默认;
2) 添加需要封装的.cpp文件,并加入对应的.h文件,先说明类的封装(也就是类的cpp文件)
头文件.h
class __declspec(dllexport) nb (类的头文件中只需修改此处即可) { public: private: }
其中关键字dllexport说明该类的实现部分需要导出。
源文件.cpp
添加一句#include "stdafx.h"即可
再说明一下一般函数的封装
将函数的定义改为extern "c" __declspec(dllexport) float add(float a, float b);
extern "c" __declspec(dllexport) float min(float a,float b);
float max(float a,float b);
函数max为导出到dll文件中,因此相当于不可见。
注意:若发现只生成了dll文件,而没有对应的lib文件,需要将cpp文件中的函数实体放到与工程同名的cpp文件下,并将原来的cpp文件删除,重新rebuild all即可。
3)编译,在debug目录下会生成对应的 .lib 和 .dll文件。
dll文件的使用
dll文件的使用分为动态加载和静态加载,两种方法优势都很明显:
动态加载:在程序中用程序显示加载dll文件,通过函数getprocaddress找到对应的函数的地址,利用函数指针直接访问对应的函数。该方法不需要lib文件的支持,也不需要包含对应的头文件,但是需要显式的加载dll文件,这也就意味着dll文件在编译的时候就已经被链接,且使用不是很方便,函数指针容易出错。
静态加载:只加载对应的lib文件(是一些函数链接的信息,一般比较小),不加载dll文件,编译的时候只需要lib文件的支持,在运行的时候才调用dll文件的函数。此处lib文件和dll文件都放在工程目录下,且需要包含对应的头文件。
windows中dll、exe都是可执行的文件,但是exe可以直接被操作系统调用执行,而dll文件不能,因此当有进程调用dll文件中函数时,直接即可运行。
动态加载过程:
1)、加载dll
2)、取函数地址
3)、释放dll
#include <windows.h> // 包含 hinstance #include <iostream> using namespace std; int main() { hinstance his = loadlibrarya("to_test1.dll");//用于加载dll typedef float(*add)(float a,float b); add add1 = (add)getprocaddress(his,"add");//getprocaddress()用于获得函数地址 typedef float(*min)(float a,float b); min add2 = (min)getprocaddress(his,"min"); cout<<add1(2,3)<<endl; cout<<add2(2,3)<<endl; freelibrary(his);//释放dll return 0; }
静态加载过程:
1)、加载lib文件:
法一:预编译命令#pragma comment(lib,"jsplayer.lib")
法二:通过资源文件加入lib文件
法三:工程属性——链接器——输入——附加依赖项
2)、包含头文件
3)、dll文件中的函数对于程序可见
#include"func_add_min.h" int main() { cout<<add(1,2)<<endl; cout<<min(1,2)<<endl; return 0; }
注意:如果是类的封装,头文件需要做简单的修改
class __declspec(dllexport) nb { public: private: }
lib文件与dll
(1)lib是编译时需要的,dll是运行时需要的。
如果要完成源代码的编译,有lib就够了。
如果也使动态连接的程序运行起来,有dll就够了。
在开发和调试阶段,当然最好都有。
(2)一般的动态库程序有lib文件和dll文件。lib文件是必须在编译期就连接到应用程序中的,而dll文件是运行期才会被调用的。如果有dll文件,那么对应的lib文件一般是一些索引信息,具体的实现在dll文件中。如果只有lib文件,那么这个lib文件是静态编译出来的,索引和实现都在其中。静态编译的lib文件有好处:给用户安装时就不需要再挂动态库了。但也有缺点,就是导致应用程序比较大,而且失去了动态库的灵活性,在版本升级时,同时要发布新的应用程序才行。
(3)在动态库的情况下,有两个文件,一个是引入库(.lib)文件,一个是dll文件,引入库文件包含被dll导出的函数的名称和位置,dll包含实际的函数和数据,应用程序使用lib文件链接到所需要使用的dll文件,库中的函数和数据并不复制到可执行文件中,因此在应用程序的可执行文件中,存放的不是被调用的函数代码,而是dll中所要调用的函数的内存地址,这样当一个或多个应用程序运行是再把程序代码和被调用的函数代码链接起来,从而节省了内存资源。从上面的说明可以看出,dll文件必须随应用程序一起发行,否则应用程序将会产生错误。
加载lib文件方法
直接加入
在vc中打开file view一页,选中工程名,单击鼠标右键,然后选中"add files to project"菜单,在弹出的文件对话框中选中要加入dll的lib文件即可。
设置工程的 project setting
打开工程的 project settings菜单,选中link,然后在object/library modules下的文本框中输入dll的lib文件。
通过程序代码
加入预编译指令#pragma comment (lib,"*.lib"),这种方法优点是可以利用条件预编译指令链接不同版本的lib文件。因为,在debug方式下,产生的lib文件是debug版本,如regd.lib;在release方式下,产生的lib文件是release版本,如regr.lib。
当应用程序对dll的lib文件加载后,还需要把dll对应的头文件(*.h)包含到其中,在这个头文件中给出了dll中定义的函数原型,然后声明。
vs2010如何生成.dll和.lib库(sdk)
1、新建一个空项目.exe或者.dll或者.lib都可以,本例子用.exe为例
新建工程--空项目,取名testdetect
2、c++文件testdetect.cpp
内容如下:_declspec(dllexport) 一定要加在函数声明和定义前面,否则在生成.dll的时候不会生成.lib(生成.dll的时候一般都会生成.lib,生成.lib的时候只有.lib)
注意:
#include "testdetec.h" _declspec(dllexport) int testdetect(short* refpcm, int reflen, int refsamprate, int refchannel, short* micpcm, int miclen, int micsamprate, int micchannel){ int resulut=0; return resulut; }
c++文件testdetect.h
#ifndef _testdetect_h_ #define _testdetect_h_ _declspec(dllexport) int testdetect(short* refpcm, int reflen, int refsamprate, int refchannel, short* micpcm, int miclen, int micsamprate, int micchannel); #endif
3、设置
如果想要生成.dll此时修改设置:属性--配置属性--常规--项目默认值--配置类型(三个选择:应用程序.exe,动态库.dll,静态库.lib等)--动态库.dll
然后buid,再按f6就可以生成动态库了,在工程目录文件夹里的dubug下面就有testdetect.dll和testdetect.lib,给别人调用时只要将testdetec.h和testdetect.dll和testdetect.lib给别人就可以了
同理可以生成.lib库