Windows 應用程序入口函數
GUI(Graphical User Interface)應用,鏈接器選項:/SUBSYSTEM:WINDOWS
CUI(Console User Interface)應用,鏈接器選項:/SUBSYSTEM:CONSOLE
_tWinMain 與 _tmain 函數聲明
Int WINAPI _tWinMain(
HINSTANCE hInstanceExe,
HINSTANCE,
PTSTR pszCmdLine,
int nCmdShow);
int _tmain(
int argc,
TCHAR *argv[],
TCHAR *envp[]);
Windows 的動態鏈接庫(Dynamic-Link Library)
部分知識點來自《Windows 核心編程(第五版)》
用處
(1)擴展了應用程序的特性
(2)簡化了項目管理
(3)有助于節省內存
(4)促進了資源的共享
(5)促進了本地化
(6)有助于解決平臺間的差異
(7)可以用于特殊目的
注意
(1)創建 DLL,事實上是在創建可供一個可執行模塊調用的函數
(2)當一個模塊提供一個內存分配函數(malloc、new)的時候,它必須同時提供另一個內存釋放函數(free、delete)
(3)在使用 C 和 C++ 混編的時候,要使用 extern "C" 修飾符
(4)一個 DLL 可以導出函數、變量(避免導出)、C++ 類(導出導入需要同編譯器,否則避免導出)
(5)DLL 模塊:cpp 文件中的 __declspec(dllexport) 寫在 include 頭文件之前
(6)調用 DLL 的可執行模塊:cpp 文件的 __declspec(dllimport) 之前不應該定義 MYLIBAPI
加載 Windows 程序的搜索順序
1、包含可執行文件的目錄
2、Windows 的系統目錄,可以通過 GetSystemDirectory 得到
3、16 位的系統目錄,即 Windows 目錄中的 System 子目錄
4、Windows 目錄,可以通過 GetWindowsDirectory 得到
5、進程的當前目錄
6、PATH 環境變量中所列出的目錄
DLL 入口函數
DllMain 函數
BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved)
{
switch(fdwReason)
{
case DLL_PROCESS_ATTACH:
// 第一次將一個DLL映射到進程地址空間時調用
// The DLL is being mapped into the process' address space.
break;
case DLL_THREAD_ATTACH:
// 當進程創建一個線程的時候,用于告訴DLL執行與線程相關的初始化(非主線程執行)
// A thread is bing created.
break;
case DLL_THREAD_DETACH:
// 系統調用 ExitThread 線程退出前,即將終止的線程通過告訴DLL執行與線程相關的清理
// A thread is exiting cleanly.
break;
case DLL_PROCESS_DETACH:
// 將一個DLL從進程的地址空間時調用
// The DLL is being unmapped from the process' address space.
break;
}
return (TRUE); // Used only for DLL_PROCESS_ATTACH
}
載入卸載庫
LoadLibrary、LoadLibraryExA、LoadPackagedLibrary、FreeLibrary、FreeLibraryAndExitThread 函數聲明
載入庫
HMODULE WINAPI LoadLibrary(
_In_ LPCTSTR lpFileName
);
HMODULE LoadLibraryExA(
LPCSTR lpLibFileName,
HANDLE hFile,
DWORD dwFlags
);
若要在通用 Windows 平臺(UWP)應用中加載 Win32 DLL,需要調用 LoadPackagedLibrary,而不是 LoadLibrary 或 LoadLibraryEx
HMODULE LoadPackagedLibrary(
LPCWSTR lpwLibFileName,
DWORD Reserved
);
卸載庫
BOOL WINAPI FreeLibrary(
_In_ HMODULE hModule
);
卸載庫和退出線程
VOID WINAPI FreeLibraryAndExitThread(
_In_ HMODULE hModule,
_In_ DWORD dwExitCode
);
顯示地鏈接到導出符號
GetProcAddress 函數聲明
FARPROC GetProcAddress(
HMODULE hInstDll,
PCSTR pszSymbolName // 只能接受 ANSI 字符串,不能是 Unicode
);
DumpBin.exe 查看 DLL 信息
在 VS 的開發人員命令提示符使用 DumpBin.exe 可查看 DLL 庫的導出段(導出的變量、函數、類名的符號)、相對虛擬地址(RVA,relative virtual address)。如:
DUMPBIN -exports D:mydll.dll
LoadLibrary 與 FreeLibrary 流程圖
LoadLibrary 與 FreeLibrary 流程圖
LoadLibrary
FreeLibrary
DLL 庫的編寫(導出一個 DLL 模塊)
DLL 庫的編寫(導出一個 DLL 模塊) DLL 頭文件
// MyLib.h
// MYLIBAPI 應該在全部 DLL 源文件的 include "Mylib.h" 之前被定義
// 全部函數/變量正在被導出
// 這個頭文件被一個exe源代碼模塊包含,意味著全部函數/變量被導入
// 這里定義任何的數據結構和符號
// 定義導出的變量(避免導出變量)
MYLIBAPI int g_nResult;
// 定義導出函數原型
MYLIBAPI int Add(int nLeft, int nRight);
DLL 源文件
// MyLibFile1.cpp
// 包含標準Windows和C運行時頭文件
// DLL源碼文件導出的函數和變量
// 包含導出的數據結構、符號、函數、變量
// 將此DLL源代碼文件的代碼放在此處
int g_nResult;
int Add(int nLeft, int nRight)
{
g_nResult = nLeft + nRight;
return g_nResult;
}
DLL 庫的使用(運行時動態鏈接 DLL)
DLL 庫的使用(運行時動態鏈接 DLL)
// A simple program that uses LoadLibrary and
// GetProcAddress to access myPuts from Myputs.dll.
typedef int (__cdecl *MYPROC)(LPWSTR);
int main( void )
{
HINSTANCE hinstLib;
MYPROC ProcAdd;
BOOL fFreeResult, fRunTimeLinkSuccess = FALSE;
// Get a handle to the DLL module.
hinstLib = LoadLibrary(TEXT("MyPuts.dll"));
// If the handle is valid, try to get the function address.
if (hinstLib != NULL)
{
ProcAdd = (MYPROC) GetProcAddress(hinstLib, "myPuts");
// If the function address is valid, call the function.
if (NULL != ProcAdd)
{
fRunTimeLinkSuccess = TRUE;
(ProcAdd) (L"Message sent to the DLL function ");
}
// Free the DLL module.
fFreeResult = FreeLibrary(hinstLib);
}
// If unable to call the DLL function, use an alternative.
if (! fRunTimeLinkSuccess)
printf("Message printed from executable ");
return 0;
}
運行庫(Runtime Library)
典型程序運行步驟
(1)操作系統創建進程,把控制權交給程序的入口(往往是運行庫中的某個入口函數)
(2)入口函數對運行庫和程序運行環境進行初始化(包括堆、I/O、線程、全局變量構造等等)。
(3)入口函數初始化后,調用 main 函數,正式開始執行程序主體部分。
(4)main 函數執行完畢后,返回到入口函數進行清理工作(包括全局變量析構、堆銷毀、關閉I/O等),然后進行系統調用結束進程。
一個程序的 I/O 指代程序與外界的交互,包括文件、管程、網絡、命令行、信號等。更廣義地講,I/O 指代操作系統理解為 “文件” 的事物。
glibc 入口
_start -> __libc_start_main -> exit -> _exit
其中 main(argc, argv, __environ)函數在__libc_start_main 里執行。
MSVC CRT 入口
int mainCRTStartup(void)
執行如下操作:
(1)初始化和 OS 版本有關的全局變量。
(2)初始化堆。
(3)初始化 I/O。
(4)獲取命令行參數和環境變量。
(5)初始化 C 庫的一些數據。
(6)調用 main 并記錄返回值。
(7)檢查錯誤并將 main 的返回值返回。
C 語言運行庫(CRT)
大致包含如下功能:
啟動與退出:包括入口函數及入口函數所依賴的其他函數等。
標準函數:有 C 語言標準規定的C語言標準庫所擁有的函數實現。
I/O:I/O 功能的封裝和實現。
堆:堆的封裝和實現。
語言實現:語言中一些特殊功能的實現。
調試:實現調試功能的代碼。
C語言標準庫(ANSI C)
包含:
標準輸入輸出(stdio.h)
文件操作(stdio.h)
字符操作(ctype.h)
字符串操作(string.h)
數學函數(math.h)
資源管理(stdlib.h)
格式轉換(stdlib.h)
時間/日期(time.h)
斷言(assert.h)
各種類型上的常數(limits.h & float.h)
變長參數(stdarg.h)
非局部跳轉(setjmp.h)
-
WINDOWS
+關注
關注
4文章
3551瀏覽量
88805 -
函數
+關注
關注
3文章
4333瀏覽量
62691 -
C++
+關注
關注
22文章
2110瀏覽量
73687 -
動態鏈接庫
+關注
關注
0文章
11瀏覽量
7066
原文標題:C++基礎語法梳理:Windows 的動態鏈接庫
文章出處:【微信號:cyuyanxuexi,微信公眾號:C語言編程學習基地】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論