lib and dll

   使用静态调用方式时,需要将在生成动态链接库时产生的lib文件添加到应用程序工程中去,对要使用的DLL中的函数只须声明一下即可,而无需调用LoadLibrary和FreeLibrary对DLL进行现式加载、卸载。这里用到lib文件是与dll文件相对应的导入文件,包含有每一个DLL导出函数的符号名和可选的标志号,但不含实际代码。LIB文件作为DLL的替换文件被编译到应用程序中。在程序员以静态调用方式编译生成应用程序时,应用程序中的调用函数将与lib文件中的导出符号相匹配,这些符号或者标志号进入新生成的可执行文件中。lib文件中包含了对应的dll文件名,链接程序将其存储在可执行文件中。一旦需要动态链接库就会根据这些信息有操作系统加载动态链接函数,所有被应用程序调用的dll文件都会在应用程序exe被加载到内存的同时被加载到内存中去。操作系统系统在加载使用可执行程序的同时加载相应的动态链接库,有可执行程序直接通过函数名来调用dll中的各个输出函数,其调用方法与调用程序内部的函数式一样的。
  在进行动态调用时,应用程序使用LoadLibrary或MFC 提供的AfxLoadLibrary显式加载动态链接库,以dll文件名作为函数的参数,在调用GetProcAddress获取想要引入的函数.就可以像使用应用程序内部函数一样调用此函数的。在退出应用程序之前,要显式的调用FreeLibrary或者MFC提供的AfxFreeLibrary显示的卸载动态链接库。
  例如: extern "C" __declspec(dllexport) double SquareRoot(double d);
       可用如下代码对其进行动态调用:
       typedef double (SQRTPROC ) (double);
       HINTSTNACE hInstance;
       SQRTPROC *pFunction;
       hInstance = ::LoadLibrary("Test.dll");
       pFunction = (SQRTPROC *)::GetProcAddress(hInstance,"SquareRoot");
       double d = (*pFunction )(81.0);

函数指针
#include <iostream
#include <string
using namespace std; 
 
int test(int a); 
 
void main(int argc,charargv[])   

    cout<<test<<endl; 
    typedef int (*fp)(int a);//注意,这里不是生命函数指针,而是定义一个函数指针的类型,这个类型是自己定义的,类型名为fp 
    fp fpi;//这里利用自己定义的类型名fp定义了一个fpi的函数指针! 
    fpi=test; 
    cout<<fpi(5)<<"|"<<(*fpi)(10)<<endl; 
    cin.get(); 

 
int test(int a) 

    return a; 
}

 

 

 

 

.dll是在你的程序运行的时候才连接的文件,因此它是一种比较小的可执行文件格式,.dll还有其他的文件格式如.ocx等,所有的.dll文件都是可执行。

 

.lib是在你的程序编译连接的时候就连接的文件,因此你必须告知编译器连接的lib文件在那里。一般来说,与动态连接文件相对比,lib文件也被称为是静态连接库。当你把代码编译成这几种格式的文件时,在以后他们就不可能再被更改。如果你想使用lib文件,就必须:
包含一个对应的头文件告知编译器lib文件里面的具体内容
设置lib文件允许编译器去查找已经编译好的二进制代码


如果你想从你的代码分离一个dll文件出来代替静态连接库,仍然需要一个lib文件。这个lib文件将被连接到程序告诉操作系统在运行的时候你想用 到什么dll文件,一般情况下,lib文件里有相应的dll文件的名字和一个指明dll输出函数入口的顺序表。如果不想用lib文件或者是没有lib 件,可以用WIN32 API函数LoadLibraryGetProcAddress。事实上,我们可以在Visual C++ IDE中以二进制形式打开lib文件,大多情况下会看到ASCII码格式的C++函数或一些重载操作的函数名字。


.h文件是在编译的时候才用到的

.lib文件是在链接的时候才用到的

.dll文件是在运行的时候才用到的

 

一般我们最主要的关于lib文件的麻烦就是出现unresolved symble 这类错误,这就是lib文件连接错误或者没有包含.c.cpp文件到工程里,关键是如果在C++工程里用了C语言写的lib文件,就必需要这样包含:

 

extern "C"
{
#include "myheader.h"
}


这是因为C语言写的lib文件没有C++所必须的名字破坏,C函数不能被重载,因此连接器会出错。

 

===============================

VC中不用MFC如何制作dll

方法一:使用export  import

VC中建立一个Console Application,建立2个文件:Dll.h  Dll.cpp

Dll.h

#ifdef MYLIBAPI
#else
#define MYLIBAPI extern "C" _declspec (dllimport)
#end if

MYLIBAPI int Add (int iLeft, int iRight) 
MYLIBAPI int Sub (int iLeft, int iRight) 

Dll.cpp

#define MYLIBAPI extern "C" _declspec (dllexport)

#include "Dll.h"

int Add (int iLeft, int iRight)
{
return iLeft + iRight ;
}

int Sub (int iLeft, int iRight)
{
return iLeft - iRight ;
}

保存文件。
Project->setting->link 最下面加上 “/dll” "/"之前一定要与前一项
有空格。
然后编译,就可以在debug  release下面找到dll  lib 文件了
使用的时候包含dll.h文件

 


方法二:使用def文件
建立一个console application, 建立2个文件dll.h  dll.cpp

Dll.h

int Add (int iLeft, int iRight) ;
int Sub (int iLeft, int iRight) ;

Dll.cpp

#include "Dll.h"

int Add (int iLeft, int iRight)
{
return iLeft + iRight ;
}

int Sub (int iLeft, int iRight)
{
return iLeft - iRight ;
}

然后再当前目录下面建立一个.def文件,文件名最好和要输出的dll名字一样,扩展名
.def, 里面写上:

LIBRARY dllname.dll
EXPORTS
Add @1
Add @2
然后将这个文件添加到工程中,
link中设置 /dll 然后编译
debugrelease中就可以找到dlllib
使用的时候加上dll.h文件

你可能感兴趣的:(dll)