动态链接库的编写与调用
動態鏈接庫是編繹好的二進制文件與exe文件類似,但它不能單獨運行。它是一個獨立的模塊,也包含了代碼、數據或資源,能被其它程序共享。多個程序調用動態鏈接庫里的同一個函數時,理論上該函數在內存中只存在一份拷貝。
靜態鏈接庫編繹器在編繹時會將函數和過程都編繹到可執行文件中,這樣會造成可執行文件很大。多個程序調用靜態鏈接庫時,每個程序都擁有該庫里函數的拷貝,若這些程序同時運行則內存中會擁有該庫中函數的多份拷貝。
動態鏈接庫的好處:
1 有利于程序的模塊化。
2 共享代碼、數據、資源。
3 程序在運行時調入代碼運行,而不是將其編繹到可執行文件中,節省了內存。動態鏈接庫調用方式:
1 隱式鏈接,即使不調用庫里面的函數,啟動時也會加載庫文件。需要把產生動態鏈接庫時產生的.lib文件加入到應用程序工程中,該文件包含了每一個dll導出函數的符號名和可選的標識號,但是并不含有實際的代碼。lib文件作為dll的替代文件被編譯到應用程序項目中。
2 顯式鏈接,是指在應用程序中用LoadLibrary或MFC提供的AfxLoadLibrary顯式的將自己所做的動態連接庫調進來,動態連接庫的文件名即是上面兩個函數的參數,再用GetProcAddress()獲取想要引入的函數。用完時調用FreeLibrary對動態庫進行卸載。頻繁的調用會犧牲一部分時間。
?
以下用一個例子來說明這兩種調用。
1 在VC6.0下新建一個WIN32 Dynamic Library工程,工程名為mydll1,然后在這個工程中建立mydll.h , mydll.cpp這兩個文件,文件的類容分別為:
mydll.h
#ifndef MYDLL_H
#define MYDLL_H
#ifdef API_DLL
#else
#define API_DLL _declspec(dllimport) //_declspec(dllimport)聲明函數是導入函數,表示此//函數是從外部導入進來了函數,一般在調用dll的程序中這樣聲明。此處這個頭文件會被//調用程序所使用。
#endif
API_DLL int add(int a,int b);
API_DLL int sub(int a,int b);
#endif
?
mydll.cpp
#include “mydll.h”
#define API_DLL _declspec(dllexport) //_declspec(dllexport)聲明函數是導出函數,表示此//函數是導出給外部程序使用的。
int add(int a,int b)
{
???????? return a+b;
}
int sub(int a,int b)
{
???????? return a-b;
}
編繹、鏈接后,在工程文件下可以發現mydll.dll, mydll.lib這兩個文件。這兩個文件在以后的隱式鏈接中會用到。
?
2 在VC6.0下再建一個WIN32 Dynamic Library工程,工程名為dll2,然后在這個工程中建立dll2.cpp這兩個文件,文件的類容為:
_declspec(dllexport) int max(int a,int b)
{
???????? return a>b?a:b;
}
_declspec(dllexport) int min(int a,int b)
{
???????? return a<b?a:b;
}
編繹鏈接后,可以在工程文件下發現dll2.dll這個文件。這個文件在接下來的工程中會被用來顯示鏈接。
3 在VC6.0下再建一個WIN32 Consol Application工程,工程名為calldll然后在這個工程中建立calldll.cpp文件,文件的類容為:
calldll.cpp
#include <stdio.h>
#include “windows.h”
#include “mydll.h”
#pragma comment(lib,”mydll.lib”)//這行可以不要,如果不要這行的話需要在//Project---setting---link---object/library中加入mydll.lib
?
void main()
{
??? //下面這8行是顯示加載dll2.dll中的函數
???????? HINSTANCE hint;
???????? hint=LoadLibrary(“dll2.dll”);????????????? //加載dll2.dll
???????? typedef int (*FUN)(int a,int b);??????????? // 自定義函數指針
???????? FUN MAX=(FUN)GetProcAddress(hint,“max”);// 獲取dll2.dll中的max函數句柄
???????? FUN MIN=(FUN)GetProcAddress(hint,“min”);// 獲取dll2.dll中的min函數句柄
???????? printf(“%d \n”,MAX(1,2));
???????? printf(“%d \n”,MIN(1,2));
???????? FreeLibrary(hint);?????? //釋放動態鏈接庫dll2.dll
??? //下面兩行是調用mydll中的add和sub函數。
???????? printf(“%d \n”,add(1,2));
???????? printf(“%d \n”,sub(1,2));
????????
}
在完成了以上三步后,還不能通過編繹第三個工程,要想通過編繹,先把第一個工程也就是mydll這個工程中的mydll.dll,mydll.lib,mydll.h這三個文件拷貝到calldll工程文件中也就是與calldll中的.cpp文件在同一目錄,然后把dll2這個文件中的dll2.dll也拷到calldll工程中,這下編繹、鏈接calldll這個工程,如果沒有錯誤的話。那么就可以直接運行了,運行過后會看到所期望的結果。
這里因為dll2.dll是顯示調用的所以只拷貝dll2.dll這個文件到calldll這個工程中就行了。
而mydll.dll是隱式調用的所以需要將mydll.lib也放到工程中去。
?
編寫此文參考了以下這個地址的內容。
http://blog.csdn.net/zieckey/article/details/1418653
?
轉載于:https://blog.51cto.com/w2qianjin/719249
總結
以上是生活随笔為你收集整理的动态链接库的编写与调用的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: ocp 042 第七章:管理方案对象
- 下一篇: 算法系列15天速成——第三天 七大经典排