WinMain是如何被调用的
WinMain函数
WinMain函数原型
Win32应用程序的入口函数为WinMain,函数原型在WinBase.h文件中:
int WINAPI WinMain ( _In_ HINSTANCE hInstance, |
![]() |
WinMain函数原型中的符号
符号 | 描述 | 其它 |
int | 返回值 | |
WINAPI | 函数调用约定 |
WINAPI宏展开: #define WINAPI __stdcall |
WinMain | 函数名 | |
HINSTANCE hInstance | 当前应用程序实例句柄 |
HINSTANCE宏展开: DECLARE_HANDLE(HINSTANCE); #define DECLARE_HANDLE(name) struct name##__{int unused;}; typedef struct name##__ *name |
HINSTANCE hPrevInstance | 没有意义。 它用于 16 位Windows,但现在始终为零 | |
LPSTR lpCmdLine | 命令行参数 | |
int nShowCmd | 窗口显示控制参数 |
程序入口
Win32编程入口为什么是WinMain呢?
C/C++语言中,编程入口为什么是main,而不是其它的呢?
CPU控制权传递
CPU是计算机的大脑,CPU读取指令后执行指令,即:控制CPU读指令的位置,就控制了CPU。
计算机加电后,CPU从BIOS指定位置读取并执行开机程序第一条指令,即,CPU被开机程序控制,开机程序获得了CPU的控制权。
开机程序获得CPU控制权后,其它程序如何获得CPU控制权呢?开机程序约定,我会去读取哪个地址的指令来执行。
其它程序把指令写入开机程序约定的地方,CPU的控制权就到了其它程序。
CPU控制权一步一步传递到操作系统。
CPU控制权到了操作系统后,其它程序想要获得控制权,就需要遵循操作系统的约定。
操作系统约定包括PE格式等,首先按PE格式生成可执行文件,如.exe。
双击或者调用进程创建函数启动程序,操作系统将可执行程序装载到内存,将下一条读取地址指向程序的入口地址,这样CPU的控制权传递到了可执行程序。
PE文件生成
要执行自定义的指令,就要遵循与操作系统的约定。
操作系统要求装载PE格式的文件,可以通过编译工具编译生成PE格式文件,如.exe。
编程语言
编译工具负责生成PE格式文件。
编译工具约定了输入的格式,即编程语言。
程序入口
C/C++编程语言的编译工具约定,可执行程序的入口默认是mainCRTStartup,该入口函数可以通过设置重新指定。
入口函数
C/C++编程语言约定了程序的入口函数,如main。程序入口(mainCRTStartup)调用入口函数main。
不论是main还是WinMain都只是一种约定,可以根据需要制定约定。
WinMain和main
C/C++语言的入口函数都是main,为什么Win32是WinMain?
程序入口
程序入口可以通过编译器指定,根据需要指定入口点,控制台默认入口为mainCRTStartup,Win32默认入口为WinMainCRTStartup,也设置为自定义函数。
main调用
main调用可以通过exe_main.cpp,exe_common.inl文件查看
mainCRTStartup()->__scrt_common_main()->__scrt_common_main_seh()->invoke_main()->main()。
WinMain调用
WinMain调用可以通过exe_winmain.cpp,exe_common.inl文件查看
WinMainCRTStartup()->__scrt_common_main()->__scrt_common_main_seh()->invoke_main()->WinMain()。
为什么是WinMain不是main?
不管是main还是WinMain都是编程语言的约定,约定内容包括返回值,调用约定,函数名,函数参数。
main和WinMain参数都包含命令行参数。
对Win32应用程序来说,主函数大概率会用到应用程序实例,即HINSTANCE hInstance参数。
为了给Win32应用程序开发提供便捷,制定了更适合Win32应用程序开发的接口约定WinMain。
应用程序实例也可以通过GetModuleHandle获取,也就是说,只要通过设置,用main或者自定义的函数作为主函数也可以编写Win32应用程序。
构建原理
入口函数缺失
在控制台程序中,不写main函数,编译报错:
1>MSVCRTD.lib(exe_main.obj) : error LNK2019: 无法解析的外部符号 main,函数 "int __cdecl invoke_main(void)" (?invoke_main@@YAHXZ) 中引用了该符号
在Win32程序中,不写WinMain函数,编译报错:
1>MSVCRTD.lib(exe_winmain.obj) : error LNK2019: 无法解析的外部符号 WinMain,函数 "int __cdecl invoke_main(void)" (?invoke_main@@YAHXZ) 中引用了该符号
LIB文件
上面的MSVCRTD.lib(exe_main.obj) 是什么意思?
.lib文件可以是静态链接库,也可以是导入库。
静态链接库包含可执行代码,导入库包含导入库信息。可以通过lib命令查看lib文件的类型。
MSVCRTD.lib是一个静态链接库。
exe_main.obj是MSVCRTD.lib中的一部分。
链接
链接时,由于程序入口为mainCRTStartup或WinMainCRTStartup,因此链接时会对程序入口的依赖进行检查。
main链接时,exe_main.obj包含mainCRTStartup()。
mainCRTStartup()->__scrt_common_main()->__scrt_common_main_seh()->invoke_main()->main()。
如果不写main函数,链接找不到main函数,链接报错。
main链接时,exe_winmain.obj包含WinMainCRTStartup()。
WinMainCRTStartup()->__scrt_common_main()->__scrt_common_main_seh()->invoke_main()->WinMain()。
如果不写WinMain函数,链接找不到WinMain函数,链接报错。