dll的加载方式主要分为两大类,显式和隐式链接

时间:2020-12-23 08:29:45

之前简单写过如何创建lib和dll文件及简单的使用(http://blog.csdn.net/betabin/article/details/7239200)。现在先再深入点写写dll的加载方式。

dll的加载方式主要分为两大类,显式和隐式链接。具体名词解释如下:

隐式链接有时称为静态加载或加载时动态链接。

显式链接有时称为动态加载或运行时动态链接。

这样我们就大概理解了这两种链接方式了,然后我们再来讲讲如何具体的操作。

在隐式链接下,使用 DLL 的可执行文件链接到该 DLL 的创建者所提供的导入库(.lib 文件)。

使用 DLL 的可执行文件加载时,操作系统加载此 DLL。客户端可执行文件调用 DLL 的导出函数,就好像这些函数包含在可执行文件内一样。

在显式链接下,使用 DLL 的可执行文件必须进行函数调用以显式加载和卸载该 DLL,并访问该 DLL 的导出函数。

客户端可执行文件必须通过函数指针调用导出函数。

一、隐式链接

1、简单的方法,是将dll对应的lib文件直接添加到项目工程里面。操作:菜单→项目→添加现有项→选择lib文件。

2、添加项目输入文件,也是将dll对应的lib文件直接添加到项目输入。操作:菜单→项目→项目属性→通用属性→链接器→输入→附加依赖项,输入lib文件并确定。

(把lib文件放至工程目录为妙。)

3、最后一种就是通过预编译指令。也就是我前面文章使用到的方法。操作:往代码中加入预编译指令如下#pragma comment (lib,”*.lib”),这样

就可以将工程目录下的所有lib包含进来。当然,要具体定位某个也行。这里只是告诉大家有这么一个指令。

最后吧,不管是哪种方法,还是需要将dll的h文件添加到项目里面,并且有引用。然后别忘了把dll放到exe目录下面了。

二、显式链接

显示链接也就这么一种方法,就是调用MS的API。大概有三个步骤:

  • 调用 LoadLibrary(或相似的函数,MFC使用AfxLoadLibrary,貌似有线程问题要处理)以加载 DLL 和获取模块句柄。

  • 调用 GetProcAddress,以获取指向应用程序要调用的每个导出函数的函数指针。由于应用程序是通过指针调用 DLL 的函数,编译器不生成外部引用,故无需与导入库链接。

  • 使用完 DLL 后调用 FreeLibrary,同样MFC使用AfxFreeLibrary

    具体的函数介绍,就不介绍了,大家自己搜下吧。下面就贴上msdn的一段示例代码吧。

    1. typedef UINT (CALLBACK* LPFNDLLFUNC1)(DWORD,UINT);
    2. ...
    3. HINSTANCE hDLL;               // Handle to DLL
    4. LPFNDLLFUNC1 lpfnDllFunc1;    // Function pointer
    5. DWORD dwParam1;
    6. UINT  uParam2, uReturnVal;
    7. hDLL = LoadLibrary("MyDLL");
    8. if (hDLL != NULL)
    9. {
    10. lpfnDllFunc1 = (LPFNDLLFUNC1)GetProcAddress(hDLL,
    11. "DLLFunc1");
    12. if (!lpfnDllFunc1)
    13. {
    14. // handle the error
    15. FreeLibrary(hDLL);
    16. return SOME_ERROR_CODE;
    17. }
    18. else
    19. {
    20. // call the function
    21. uReturnVal = lpfnDllFunc1(dwParam1, uParam2);
    22. }
    23. }
    typedef UINT (CALLBACK* LPFNDLLFUNC1)(DWORD,UINT);
    ... HINSTANCE hDLL; // Handle to DLL
    LPFNDLLFUNC1 lpfnDllFunc1; // Function pointer
    DWORD dwParam1;
    UINT uParam2, uReturnVal; hDLL = LoadLibrary("MyDLL");
    if (hDLL != NULL)
    {
    lpfnDllFunc1 = (LPFNDLLFUNC1)GetProcAddress(hDLL,
    "DLLFunc1");
    if (!lpfnDllFunc1)
    {
    // handle the error
    FreeLibrary(hDLL);
    return SOME_ERROR_CODE;
    }
    else
    {
    // call the function
    uReturnVal = lpfnDllFunc1(dwParam1, uParam2);
    }
    }