【VC++】字符串详解&窗口&第一个windows程序

注:最后有面试挑战,看看自己掌握了吗

系统调用顺序对比

在这里插入图片描述

  1. windows没开放源码,所以要在封装一层API,不是像 Linux 那样直接使用系统调用
  2. 应用程序在 Linux 和 Windows 都是用 fwrite 函数,两个操作系统都会调用运行时库中的 write 函数。
  3. 二者区别是二者运行时库的名字不同。Linux中 write 函数在 libc.so 动态库中实现的,而 Windows 则是在 msvcr90.dll 动态库中实现的。

在这里插入图片描述
根本原因是程序使用了标准函数,而在你的机器上,没有这个标准函数对应的动态库实现,如果想要顺利运行则需要到官方网站下载对应版本的动态库安装即可

怎样避免确实动态链接库

在这里插入图片描述
如果你没有依赖其它第三方库,默认 IDE 中指定的运行时库的方式是动态链接,这情况下,会依赖于特定版本的运行时库,具体的编译选项是
/MT、/MTd 以及 /MD 和 /MDd 。后面的小d表示 Debug 和 Release 之分,而 MT 和 MD
则是选择使用静态编译还是动态编译,如果发布程序选择 /MT 选项进行链接,则可以避免在其它机器上运行时缺失动态库的麻烦。

基本知识

类型列表

  1. Windows 头文件中包含了很多独特的类型定义,大多数的定义都在 WinDef.h 文件中,下面是常用的类型列表:

数据类型 大小 有符号?
BYTE 8 bits Unsigned
DWORD 32 bits Unsigned
INT32 32 bits Signed
INT64 64 bits Signed
LONG 32 bits Signed
LONGLONG 64 bits Signed
UINT32 32 bits Unsigned
UINT64 64 bits Unsigned
ULONG 32 bits Unsigned
ULONGLONG 64 bits Unsigned
WORD 16 bits Unsigned

  1. 为了兼容一些历史的程序,所以列表中的类型存在一定的冗余
  2. 这些类型在32位和64位程序中的大小都是固定的。
  3. DWORD 类型在32位和64位程序中都是32位大小。

指针类型

  1. Windows 定义了很多指针类型,通常都是以 P- 或者 LP- 开始的类型名称。
  2. LPRECT 代表指向 RECT 类型的指针。 RECT 类型代表一个矩形结构体, 下面几种定义方式是等价的:

RECT* rect; // Pointer to a RECT structure.
LPRECT rect; // The same PRECT rect; // Also the same.

  1. 历史上,P 前缀的指针叫 pointer ,LP 前缀的指针叫 long pointer。
  2. 二者是16位操作系统向32位操作系统过渡的产物,如今已经没有区别了。 虽然如今已经不分“长短”了,但是依然会出现一个兼容问题,特别是在程序逻辑中存在指针和整形的转化关系的时候。为了解决这个问题,Windows 定义了几种明确的指针类型

DWORD_PTR
INT_PTR
LONG_PTR
ULONG_PTR
UINT_PTR

  1. 如果在程序逻辑上存在32位值扩展到64位值的情况,使用上述类型。

匈牙利标记法

  1. 匈牙利标记法这是为了纪念具有传奇色彩的微软程序员 Charles Simonyi。这种标记法非常简单,即变量名表明该变量数据类型的小写字母开始。
  2. szCmdLine 的前缀 sz 表示“以零结束的字符串”。

p 指针
h 句柄
sz 字符串
b BYTE
ul ULONG
l LONG
i INT
ui UINT
ch CHAR
dw DWORD
hr HRESULT

字符串详解

在这里插入图片描述

  1. Windows天然支持 Unicode 字符串。
  2. 字符串经常被用在 UI 组件、文件名等字符相关的地方。
  3. 因为 Windows 操作系统会涉及到多语言的问题,所以 Unicode 是首选的字符串编码方式。
  4. Linux 使用 UTF-8 编码而Windows 平台使用 UTF-16 编码方式
  5. 每个字符用16位的值表示。UTF-16 字符也被叫做宽字符
  6. Visual Studio C++ 编译器支持内置的宽字符类型 wchar_t ,具体定义在头文件 WinNT.h 中。
  7. typedef wchar_t WCHAR声明一个宽字符或者一个宽字符串需要将 L 放到文件前面:
wchar_t a = L'a';
wchar_t *str = L"hello";
  1. 常见的字符串类型:
类型	定义
CHAR	char
PSTR or LPSTR	char*
PCSTR or LPCSTR	const char*
PWSTR or LPWSTR	wchar_t*
PCWSTR or LPCWSTR	const wchar_t*

Unicode 和 ANSI 函数

因为微软提供对 Unicde 的支持,所以它将每个和字符串相关的 API 都提供了两个版本,一种是 ANSI 字符串版本,另一种是Unicode 字符串版本。
在这里插入图片描述
为使计算机支持更多语言,通常使用 0x80~0xFFFF 范围的 2 个字节表示 1 个字符。比如:汉字 ‘中’ 在中文操作系统中,使用 [0xD6,0xD0] 这两个字节存储。
不同的国家和地区制定了不同的标准,由此产生了 GB2312、GBK、GB18030、Big5、Shift_JIS 等各自的编码标准。这些使用多个字节来代表一个字符的各种汉字延伸编码方式,称为 ANSI 编码。在简体中文Windows操作系统中,ANSI 编码代表 GB2312编码;在繁体中文Windows操作系统中,ANSI编码代表Big5;在日文Windows操作系统中,ANSI 编码代表 JIS 编码。 [1] [2]
不同 ANSI 编码之间互不兼容,当信息在国际间交流时,无法将属于两种语言的文字,存储在同一段 ANSI 编码的文本中。ANSI编码表示英文字符时用一个字节,表示中文用两个或四个字节。
ANSI编码作为中国以及部分亚太地区的多字符编码格式Windows系统和OS X都是提供原生支持的。但是即便如此,许多国外开发者仍然在开发笔记或者文字录入类应用的时候将ANSI编码完全忽略只加入全球通用的UTF-8编码

  1. 下面两个 API 都是设置窗口标题的接口:
    SetWindowTextA 需要传入 ANSI 字符串。
    SetWindowTextW 需要传入 Unicode 字符串。
  2. 函数内部,ANSI 版本的接口会将 ANSI 字符串转换为Unicode 字符串,然后再调用 Unicode 版本接口完成操作。
  3. 为了方便微软在头文件中还定义了一个 UNICODE 宏来区分不同版本的调用
#ifdef UNICODE
#define SetWindowText  SetWindowTextW
#else
#define SetWindowText  SetWindowTextA
#endif
  1. 使用上面的方法可以避免在代码中区分具体调用的接口是 UNICODE 版本还是 ANSI 版本。相反的,在程序中只需要直接调用 SetWindowText 函数设置标题即可。
  2. 编译的时候会自动根据 UNICODE 宏定义决定使用哪个版本。
  3. 除非兼容旧的程序或者组件,否则在开发新应用的时候,应该一直使用 Unicode 版本。因为 Windows 操作系统支持多语言,如果使用 ANSI 版本,它将无法支持应用程序的本地化
  4. ANSI 版本的接口效率更低,因为其内部需要进行编码转换

TCHARs

  1. 如果你的程序需要同时支持多款操作系统,如:Windows NT、Windows 95、Windows 98 和 Windows Me。
  2. 这时候需要明确区分使用的是 ANSI 版本还是 Unicode 版本字符串,为了进一步方面开发,Windows 提供一个宏来完成二者的自动区分
宏	           Unicode	       ANSI
TCHAR	       wchar_t	       char
TEXT(“x”)	   L”x”            “x”
  1. 例子:
SetWindowText(TEXT("My Application"));

该语句等价于:

SetWindowTextW(L"My Application"); // Unicode function with wide-character string.
SetWindowTextA("My Application");  // ANSI function.
  1. 如今TEXT 和 TCHAR 宏用处已经很小了,因为所有的程序都应该使用 Unicode 字符,然而你在一些老的程序中仍然看见它们的身影。

  2. 除了上边的问题,在头文件中,微软 C 运行时库中,仍然存在类似的宏定义,例如涉及到字符操作的函数:

#ifdef _UNICODE
#define _tcslen     wcslen
#else
#define _tcslen     strlen
#endif
  1. 一些头文件使用 UNICODE 宏,另一些使用 _UNICODE 宏,最好同时定义它们,如果你是用 Visual C++ 创建工程,这些会默认自动进行设置

窗口

  1. 存在 UI 控件的时候,控件窗口被认为是应用程序窗口的子窗口
  2. 应用程序窗口被认为是控件窗口的父窗口
  3. 通过父窗口的坐标系可以定位子窗口的位置,并且子窗口的样式等一些属性会受到父窗口的影响。例如,超出父窗口范围的子窗口默认会被裁剪掉。
    在这里插入图片描述

WinMain

  1. 每个 Windows 程序都有一个类似 main 函数的入口函数名字叫做 WinMain 或者 wWinMain , 其声明如下:int WINAPI wWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, PWSTR pCmdLine, int nCmdShow);

  2. 该函数有四个参数:
    hInstance 参数的含义是一个实例句柄,或者是一个模块句柄。该参数代表被加载到内存的可执行程序(exe文件)。某些 Windows API 可能需要该参数。如:加载图标或者加载位图等等。
    hPrevInstance参数无意义,它在过去的16位操作系统中被使用,现在永远为0。
    pCmdLine 该参数是程序传入的命令行参数,和 main 函数中的 argv 参数含义类似,只不过现在的字符集是 Unicode。
    nCmdShow 该参数是一个整数,代表应用程序主窗口的显示状态,是最小化、最大化还是正常显示。

  3. WINAPI 是一种调用约定。这个约定确保函数从调用者那里接收参数的相关规则,例如,参数在栈中保存的顺序等等。在写代码的时候一定不要忘记该约定的标识。

  4. WinMain 和 wWinMain 默认是等价的,除非命令行参数是一个 ANSI 字符串,Unicode 版本永远是首选如果是 Unicode 版本,你可以传递一个 ANSI 字符串,反之不可

  5. 可以通过 GetCommandLine 函数获取命令行参数,这个函数返回一个单一的字符串。

  6. 获取参数的数组样式,可以通过 CommandLineToArgvW 函数完成。

  7. 编译器是怎样知道调用 wWinMain 还是调用标准的 main 函数?
    实际情况下,在微软的 C 运行时库(CRT)中提供了一个 main 函数实现,其内部会调用 WinMain 或者 wWinMain
    CRT 中 main 函数的内部在调用 wWinMain 之前做了一些额外的工作
    例如初始化一些静态成员或者其它 C 函数的初始化操作等等。虽然你可以手动指定不同的链接不同的入口函数,但是仍然推荐你使用 CRT 默认提供的入口点函数,否则 CRT 内部的一些代码将被会跳过,有可能会导致一些异常的结果。

  8. 空的 WinMain 函数:

INT WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,
    PSTR lpCmdLine, INT nCmdShow)
{
    return 0;
}

🍃博主昵称:一拳必胜客
🌸博主寄语:欢迎点赞收藏关注哦,一起成为朋友一起成长;

 特别鸣谢:木芯工作室
特别鸣谢:木芯工作室 、Ivan from Russia
部分来源:https://meishizaolunzi.com/winmain-ying-yong-cheng-xu-ru-kou-dian/

我的博客即将同步至腾讯云开发者社区,邀请大家一同入驻:https://cloud.tencent.com/developer/support-plan?invite_code=1noqb14tvyjrm

  • 3
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

扯淡散人

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值