动态链接库dll的 静态加载 与 动态加载

 
动态链接是指在生成可执行文件时不将所有程序用到的函数链接到一个文件,因为有许多函数在操作系统带的dll文件中,当程序运行时直接从操作系统中找。  
而静态链接就是把所有用到的函数全部链接到exe文件中。
动态链接是只建立一个引用的接口,而真正的代码和数据存放在另外的可执行模块中,在运行时再装入;  
而静态链接是把所有的代码和数据都复制到本模块中,运行时就不再需要库了。
 

静态链接方法:#pragma comment(lib, "test.lib") ,静态链接的时候,载入代码就会把程序会用到的动态代码或动态代码的地址确定下来
静态库的链接可以使用静态链接,动态链接库也可以使用这种方法链接导入库

动态链接方法:LoadLibrary()/GetProcessAddress()和FreeLibrary(),使用这种方式的程序并不在一开始就完成动态链接,而是直到真正调用动态库代码时,载入程序才计算(被调用的那部分)动态代码的逻辑地址,然后等到某个时候,程序又需要调用另外某块动态代码时,载入程序又去计算这部分代码的逻辑地址,所以,这种方式使程序初始化时间较短,但运行期间的性能比不上静态链接的程序。

 
1.生成  静态链接库 newdll)  win32项目 ->  dll

添加.h文件 
betabinlib.h

#ifndef BETABINLIB_H  
#define BETABINLIB_H  
   
#ifdef NEWDLL_EXPORTS   //自动添加的宏   右键工程-属性-配置属性-预处理器-..定义  
#define MYDLL_API extern "C" __declspec(dllexport)  
#else  
#define MYDLL_API extern "C" __declspec(dllimport)  
#endif  
   
MYDLL_API int add(int x, int y);  // 必须加前缀  
#endif  

 

添加.cpp文件  betabinlib.cpp

 

#include "stdafx.h"
#include "betabinlib.h"
 
int add(int x, int y)
{
    return x + y;
}

 

编译生成  .dll 和 .lib(1)dll的静态加载--将整个dll文件 加载到  .exe文件中
特点:程序较大,占用内存较大,但速度较快(免去 调用函数LOAD
 
#include <stdio.h>
#include "betabinlib.h"
#include <Windows.h>
#pragma comment(lib, "newdll.lib")
 
int main()
{
    printf("2 + 3 = %d \n", add(2, 3));
    return 0;
}

 

#include <stdio.h>  
#include <Windows.h>  
   
int main()  
{  
    HINSTANCE h=LoadLibraryA("newdll.dll");  
    typedef int (* FunPtr)(int a,int b);//定义函数指针  
   
    if(h == NULL)  
    {  
    FreeLibrary(h);  
    printf("load lib error\n");  
    }  
    else  
    {  
        FunPtr funPtr = (FunPtr)GetProcAddress(h,"add");  
        if(funPtr != NULL)  
        {  
            int result = funPtr(3, 3);  
            printf("3 + 3 = %d \n", result);  
        }  
        else  
        {  
            printf("get process error\n");  
            printf("%d",GetLastError());  
        }  
        FreeLibrary(h);  
    }  
   
    return 0;  
}  

 

**************************************

静态库链接时搜索路径顺序:

1. ld会去找GCC命令中的参数-L
2. 再找gcc的环境变量LIBRARY_PATH
3. 再找内定目录 /lib /usr/lib /usr/local/lib 这是当初compile gcc时写在程序内的

动态链接时、执行时搜索路径顺序:

1. 编译目标代码时指定的动态库搜索路径
2. 环境变量LD_LIBRARY_PATH指定的动态库搜索路径
3. 配置文件/etc/ld.so.conf中指定的动态库搜索路径
4. 默认的动态库搜索路径/lib
5. 默认的动态库搜索路径/usr/lib

有关环境变量:
LIBRARY_PATH环境变量:指定程序静态链接库文件搜索路径
LD_LIBRARY_PATH环境变量:指定程序动态链接库文件搜索路径

**************************************

Linux下编译链接或运行c/c++程序时可能会遇到找不到头文件,找不到库文件的错误,简单总结一下这些错误的解决方法

1,找不到头文件

解决方法一:在编译时使用 -I 来指定头文件的路径,例如把头文件放在 /home/user/include/ 目录下,则在编译时加上 -I /home/user/include/ ,如有多个目录,可多次使用-I来指定

解决方法二:将头文件的路径加入环境变量CPATH中,如 export CPATH=" /home/user/include/",也可以根据情况使用下面的三个环境变量:

C_INCLUDE_PATH 编译C程序时使用的环境变量,用于查找头文件。
CPLUS_INCLUDE_PATH 编译C++程序时使用的环境变量,用于查找头文件。
OBJC_INCLUDE_PATH 编译Obj-C程序时使用的环境变量,用于查找头文件。
CPATH 编译C/C++/Obj-C程序时使用的环境变量,用于查找头文件。

2,链接时找不到库文件

解决方法一:在链接时使用-L参数来指定库的路径,例如把某个自己制作的动态库放在/home/user/lib/目录下了,则链接程序时加上-L/home/user/lib/,同时可以使用-l来指定库的名称,如指定线程库:-lpthread

解决方法二:把库文件放入系统的库文件目录下,如/lib,/usr/lib等,操作系统运行该程序时会自动到这些目录下找库文件

解决方法三:把库文件所在的目录加入LIBRARY_PATH环境变量中,如 export LIBRARY_PATH=" /home/user/lib/"

3,运行时找不到动态库

解决方法一:把库文件放入系统的库文件目录下,如/lib,/usr/lib等

解决方法二:把库文件所在的目录加入LD_LIBRARY_PATH环境变量中,如 export LD_LIBRARY_PATH=" /home/user/lib/"

 

原文地址:https://www.cnblogs.com/loanhicks/p/7413996.html

 https://www.cnblogs.com/dirge/p/6443317.html

posted @ 2019-06-05 14:31  狂奔~  阅读(8811)  评论(0编辑  收藏  举报