visual studio静态,动态链接库开发工具简单使用
文章目录
这是2011年8月份做过的一点实验,查了MSDN等等很多资料,基本搞明白了。
这里我不会使用visual studio的图形界面工具,作为专业人士,还是搞懂自己的工具是怎么运转的,这样比较好。
要使用的是visual studio的命令行工具,其实和gcc那堆工具对应关系挺明显的,大致如下:
gcc | cl |
---|---|
ar | lib |
ld | link |
文件后缀对应关系:
gcc | cl |
---|---|
.a | .lib |
.so | .dll |
.o | .obj |
新手注意:这个命令提示符是"开始菜单"–»“Microsoft Visual studio”–»“Visual Tools” 那里的提示符,其实就是设置过一些环境变量的cmd。
先把实验代码贴出来(可以先跳过不看),共有五个文件:
1.build.bat
|
|
2.库的头文件,ext.h
|
|
3.库的实现文件,ext.c
|
|
4.以动态链接库方式使用这个库的程序代码,main.c
|
|
5.以静态链接库方式使用这个库的程序代码,main-static.c
|
|
一,静态链接库 要给用户提供静态链接库,都要提供哪些文件呢?.h头文件+.lib库文件 先问个问题: 使用库的最简单方式是什么?当然是不使用库! 不使用库谁不会啊?使用如下命令:
|
|
在当前目录下,就会编译,链接成一个main-static.exe,执行一下,没有问题!
再问个问题:除了exe文件,还看见了什么?有ext.obj, main-static.obj, main-static.lib ,main-static.exp .
.obj文件是目标文件(object file), .lib文件是静态库文件(msdn中称为导入库), .exp文件,导出文件,辅助作用(参看这里和这里) 可以猜想,用cl直接编译生成exe的过程,是
- ext.c—»ext.obj
- main-static.c–»main-static.obj
- main-static.obj—»main-static.exp+main-static.lib
- ext.obj+main-static.lib+main-static.exp—»main-static.exe
的过程
我尝试了改换参数顺序,即用命令
|
|
发现main-static.exp + main-static.lib换成了ext.lib+ext.exp 这说明是cl是先把.c文件都生成为obj文件,然后把第一个命令行参数指定xxx.c的xxx.obj文件再生成为xxx.lib+xxx.exp文件 这和静态链接有什么关系呢?关系密切啊! 把除了ext.lib 以外的所有生成文件删除,运行如下命令:
|
|
就可以看见,生成了main-static.exe,运行正常。ext.lib就是静态链接库。 其实,这就是一个静态链接的过程。
静态链接的第一步,怎么由ext.c生成obj文件?查msdn(或者 命令cl /?)可知,给cl提供/c参数就可以了,先清掉那堆文件,运行
|
|
成功生成ext.obj.同理可以生成main-static.obj。 严格来说编译已经完成,接下来的,都是链接过程。
第二步,怎么从obj文件生成.lib文件(加/DEF参数,就可以同时生成.exp文件)
|
|
就生成了ext.lib 第三步,怎么从main-static.c + ext.lib 生成exe
|
|
需要注意的是,头文件一定要可以找到。
二,动态链接库 visual studio的动态链接分为显式链接和隐式链接两种(http://msdn.microsoft.com/zh-cn/library/253b8k2c(v=vs.80).aspx),显式链接就是自己写代码调用win32 Api加载dll文件,要使用LoadLibrary, GetProcAddress(), FreeLibrary()这些函数,这个方法几乎不需要工具,就不说了,看看msdn的例子就明白了。 以下说的都是隐式链接
visual studio的动态链接库使用很与众不同,竟然需要提供三种文件:.h+.lib+.dll文件,诡异的是,这个.lib文件和静态链接库里的.lib文件其实不一样! 更诡异的是,用户链接的时候,其实不需要.dll文件!
对比之下,linux下的gcc,若是动态链接库,需要的是.so+.h 文件(.so相当于.dll),这是显著的差异。
参看csdn(http://msdn.microsoft.com/zh-cn/library/3y1sfaz2(v=vs.100).aspx)可知,和静态链接不同的是,动态链接需要对库的源代码进行改动,添加
|
|
或
|
|
补充:也可以用def文件,不过我没试过,参看 http://msdn.microsoft.com/zh-cn/library/0b9xe492.aspx 和 http://msdn.microsoft.com/zh-cn/library/28d6s79h.aspx 。
这个时候要注意,__declspec( dllimport )和__declspec( dllexport ) 的使用是不一样的,在库里导出的符号(函数,变量等,可以用dumpbin查看)应该使用__declspec( dllexport ) ,而使用库的代码,在包含头文件后,要有这些符号的声明,要看到__declspec( dllimport )形式的声明,所以这是有区别的。在上面ext.h里,我用了一个宏Dll来统一处理。
处理完源代码,就可以编译了。
第一步,生成obj文件,
|
|
第二步,由obj文件生成dll文件+lib文件,
|
|
或者也可以用
|
|
可以看到,生成了ext.dll , ext.exp, ext.lib 三个文件,其中的ext.lib 这个文件和静态链接时生成的ext.lib文件是不一样的!而且ext.dll在main.c的链接过程中是用不着的! (此时已经生成了dll文件,要是使用显式链接,就把dll文件弄走,自己写代码去啦),继续隐式链接 第三步,不妨移走ext.dll,然后编译
|
|
可以看到生成main.exe,但是运行一下,会报告找不到ext.dll,当然找不到啦,只要把ext.dll移到当前目录下(或在dll文件查找路径中的一个中)即可正常运行。
注:dll文件查找路径参看 http://msdn.microsoft.com/zh-cn/library/7d83bc18(v=vs.80).aspx
这个领域关注的人少,鄙人理解浅薄,错误难免,欢迎指正。