Tech Ideas

C++,Linux,Algorithm,Crypto,Lisp,etc

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

""C:Program Files (x86)Microsoft Visual Studio 10.0VCvcvarsall.bat"" x86
cd dynamic
del /f /Q * 
cl /LD ..ext.c 
del ext.obj 
cl ..main.c /link ext.lib
del main.obj
cd ..
cd static
del /f /Q *
cl /c ..ext.c
lib ext.obj
del ext.obj
cl ..main-static.c /link ext.lib
del main*.obj main*.exp main*.lib
cd ..

2.库的头文件,ext.h

#ifndef EXT_H
#define EXT_H
#ifdef Import
#define Dll __declspec(dllimport)
#else
#ifdef Export
#define Dll __declspec(dllexport)
#else
#define Dll
#endif
#endif

#include 
Dll typedef struct {
int i;
char c;
} st;
Dll extern int num;
Dll void fun();
Dll extern st s;
#endif
3.库的实现文件,ext.c
#define Export
#include "ext.h"
st s={24,'h'};
int num=0;
void fun(){
printf("hello,i am in lib. num=%dn",num++);
}
4.以动态链接库方式使用这个库的程序代码,main.c
#define Import
#include "ext.h" 
#include "windows.h"
#include 

int main(){
    fun();
    printf("num=%d\n",num);
    fun();
    printf("num=%d\n",num);
    printf("st i=%d c=%c \n",s.i,s.c);
}

5.以静态链接库方式使用这个库的程序代码,main-static.c

 
#include "ext.h" 
#include "windows.h"
#include

int main(){
    fun();
    printf("num=%d\n",num);
    fun();
    printf("num=%d\n",num);
    printf("st i=%d c=%c \n",s.i,s.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
         的过程
我尝试了改换参数顺序,即用命令 cl ext.c main-static.c
发现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文件)
lib ext.obj
就生成了ext.lib
第三步,怎么从main-static.c + ext.lib 生成exe
cl main-static.c /link ext.lib

需要注意的是,头文件一定要可以找到。

二,动态链接库

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 )
补充:也可以用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文件,
cl /c ext.c

第二步,由obj文件生成dll文件+lib文件,

cl /LD ext.obj

或者也可以用

link /DLL ext.obj

可以看到,生成了ext.dll , ext.exp, ext.lib 三个文件,其中的ext.lib 这个文件和静态链接时生成的ext.lib文件是不一样的!而且ext.dll在main.c的链接过程中是用不着的!

(此时已经生成了dll文件,要是使用显式链接,就把dll文件弄走,自己写代码去啦),继续隐式链接
第三步,不妨移走ext.dll,然后编译
cl main.c /link ext.lib

 

可以看到生成main.exe,但是运行一下,会报告找不到ext.dll,当然找不到啦,只要把ext.dll移到当前目录下(或在dll文件查找路径中的一个中)即可正常运行。

   注:dll文件查找路径参看 http://msdn.microsoft.com/zh-cn/library/7d83bc18(v=vs.80).aspx

这个领域关注的人少,鄙人理解浅薄,错误难免,欢迎指正。