visual studio静态,动态链接库开发工具简单使用

这是2011年8月份做过的一点实验,查了MSDN等等很多资料,基本搞明白了。这里我不会使用visual studio的图形界面工具,作为专业人士,还是搞懂自己的工具是怎么运转的,这样比较好。

要使用的是visual studio的命令行工具,其实和gcc那堆工具对应关系挺明显的,大致如下:

gcc cl
ar lib
ld link
文件后缀对应关系:
.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库文件 
先问个问题: 使用库的最简单方式是什么?当然是不使用库!
不使用库谁不会啊?使用如下命令:

        cl main-static.c ext.c 
        在当前目录下,就会编译,链接成一个main-static.exe,执行一下,没有问题!

        再问个问题:除了exe文件,还看见了什么?有ext.obj, main-static.obj, main-static.lib ,main-static.exp .
         .obj文件是目标文件(object file),
         .lib文件是静态库文件(msdn中称为导入库),
          .exp文件,导出文件,辅助作用(参看这里这里
          可以猜想,用cl直接编译生成exe的过程,是
         1. ext.c--->>ext.obj
         2.main-static.c-->>main-static.obj
         3.main-static.obj--->>main-static.exp+main-static.lib
         4.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 以外的所有生成文件删除,运行如下命令:
cl main-static.c /link ext.lib
就可以看见,生成了main-static.exe,运行正常。ext.lib就是静态链接库。

其实,这就是一个静态链接的过程。 


 静态链接的第一步,怎么由ext.c生成obj文件?查msdn(或者 命令cl /?)可知,给cl提供/c参数就可以了,先清掉那堆文件,运行

         cl /c ext.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)可知,和静态链接不同的是,动态链接需要对库的源代码进行改动,添加
      __declspec( dllimport )
 或 __declspec( dllexport ) 
这个时候要注意,__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文件查找路径中的一个中)即可正常运行。
这个领域关注的人少,鄙人理解浅薄,错误难免,欢迎指正。

原文地址:https://www.cnblogs.com/windydays/p/2298103.html