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

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

系统调用顺序对比


1668341882289.jpg


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


1668341975872.jpg

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


怎样避免确实动态链接库


1668341998962.jpg


如果你没有依赖其它第三方库,默认 IDE 中指定的运行时库的方式是动态链接,这情况下,会依赖于特定版本的运行时库,具体的编译选项是

/MT、/MTd 以及 /MD 和 /MDd 。后面的小d表示 Debug 和 Release 之分,而 MT 和 MD

则是选择使用静态编译还是动态编译,如果发布程序选择 /MT 选项进行链接,则可以避免在其它机器上运行时缺失动态库的麻烦。


基本知识


类型列表


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


为了兼容一些历史的程序,所以列表中的类型存在一定的冗余。

这些类型在32位和64位程序中的大小都是固定的。

DWORD 类型在32位和64位程序中都是32位大小。


指针类型


Windows 定义了很多指针类型,通常都是以 P- 或者 LP- 开始的类型名称。

LPRECT 代表指向 RECT 类型的指针。 RECT 类型代表一个矩形结构体, 下面几种定义方式是等价的:


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


历史上,P 前缀的指针叫 pointer ,LP 前缀的指针叫 long pointer。

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

DWORD_PTR
INT_PTR
LONG_PTR
ULONG_PTR
UINT_PTR


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


匈牙利标记法


匈牙利标记法这是为了纪念具有传奇色彩的微软程序员 Charles Simonyi。这种标记法非常简单,即变量名表明该变量数据类型的小写字母开始。

szCmdLine 的前缀 sz 表示“以零结束的字符串”。

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


字符串详解


1668342109157.jpg


Windows天然支持 Unicode 字符串。

字符串经常被用在 UI 组件、文件名等字符相关的地方。

因为 Windows 操作系统会涉及到多语言的问题,所以 Unicode 是首选的字符串编码方式。

Linux 使用 UTF-8 编码而Windows 平台使用 UTF-16 编码方式

每个字符用16位的值表示。UTF-16 字符也被叫做宽字符。

Visual Studio C++ 编译器支持内置的宽字符类型 wchar_t ,具体定义在头文件 WinNT.h 中。

typedef wchar_t WCHAR声明一个宽字符或者一个宽字符串需要将 L 放到文件前面:

wchar_t a = L'a';
wchar_t *str = L"hello";


常见的字符串类型:
类型  定义
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 字符串版本。

1668342154409.jpg

为使计算机支持更多语言,通常使用 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编码。


下面两个 API 都是设置窗口标题的接口:

SetWindowTextA 需要传入 ANSI 字符串。

SetWindowTextW 需要传入 Unicode 字符串。

在函数内部,ANSI 版本的接口会将 ANSI 字符串转换为Unicode 字符串,然后再调用 Unicode 版本接口完成操作。

为了方便微软在头文件中还定义了一个 UNICODE 宏来区分不同版本的调用。

#ifdef UNICODE
#define SetWindowText  SetWindowTextW
#else
#define SetWindowText  SetWindowTextA
#endif


使用上面的方法可以避免在代码中区分具体调用的接口是 UNICODE 版本还是 ANSI 版本。相反的,在程序中只需要直接调用 SetWindowText 函数设置标题即可。

编译的时候会自动根据 UNICODE 宏定义来决定使用哪个版本。

除非兼容旧的程序或者组件,否则在开发新应用的时候,应该一直使用 Unicode 版本。因为 Windows 操作系统支持多语言,如果使用 ANSI 版本,它将无法支持应用程序的本地化

ANSI 版本的接口效率更低,因为其内部需要进行编码转换。


TCHARs


如果你的程序需要同时支持多款操作系统,如:Windows NT、Windows 95、Windows 98 和 Windows Me。

这时候需要明确区分使用的是 ANSI 版本还是 Unicode 版本字符串,为了进一步方面开发,Windows 提供一个宏来完成二者的自动区分。

宏            Unicode        ANSI
TCHAR        wchar_t        char
TEXT(“x”)    L”x”            “x”


例子:


SetWindowText(TEXT("My Application"));

该语句等价于:


SetWindowTextW(L"My Application"); // Unicode function with wide-character string.
SetWindowTextA("My Application");  // ANSI function.


如今,TEXT 和 TCHAR 宏的用处已经很小了,因为所有的程序都应该使用 Unicode 字符,然而你在一些老的程序中仍然看见它们的身影。


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

#ifdef _UNICODE
#define _tcslen     wcslen
#else
#define _tcslen     strlen
#endif


一些头文件使用 UNICODE 宏,另一些使用 _UNICODE 宏,最好同时定义它们,如果你是用 Visual C++ 创建工程,这些会默认自动进行设置。


窗口


存在 UI 控件的时候,控件窗口被认为是应用程序窗口的子窗口,

应用程序窗口被认为是控件窗口的父窗口。

通过父窗口的坐标系可以定位子窗口的位置,并且子窗口的样式等一些属性会受到父窗口的影响。例如,超出父窗口范围的子窗口默认会被裁剪掉。


1668342232105.jpg


WinMain


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


该函数有四个参数:

hInstance 参数的含义是一个实例句柄,或者是一个模块句柄。该参数代表被加载到内存的可执行程序(exe文件)。某些 Windows API 可能需要该参数。如:加载图标或者加载位图等等。

hPrevInstance 该参数无意义,它在过去的16位操作系统中被使用,现在永远为0。

pCmdLine 该参数是程序传入的命令行参数,和 main 函数中的 argv 参数含义类似,只不过现在的字符集是 Unicode。

nCmdShow 该参数是一个整数,代表应用程序主窗口的显示状态,是最小化、最大化还是正常显示。


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


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


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


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


编译器是怎样知道调用 wWinMain 还是调用标准的 main 函数?

实际情况下,在微软的 C 运行时库(CRT)中提供了一个 main 函数实现,其内部会调用 WinMain 或者 wWinMain。

CRT 中 main 函数的内部在调用 wWinMain 之前做了一些额外的工作,

例如初始化一些静态成员或者其它 C 函数的初始化操作等等。虽然你可以手动指定不同的链接不同的入口函数,但是仍然推荐你使用 CRT 默认提供的入口点函数,否则 CRT 内部的一些代码将被会跳过,有可能会导致一些异常的结果。


空的 WinMain 函数:

INT WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,
    PSTR lpCmdLine, INT nCmdShow)
{
    return 0;
}
相关文章
|
4月前
|
C++
C++ 根据程序运行的时间和cpu频率来计算在另外的cpu上运行所花的时间
C++ 根据程序运行的时间和cpu频率来计算在另外的cpu上运行所花的时间
50 0
|
2月前
|
存储 程序员 编译器
简述 C、C++程序编译的内存分配情况
在C和C++程序编译过程中,内存被划分为几个区域进行分配:代码区存储常量和执行指令;全局/静态变量区存放全局变量及静态变量;栈区管理函数参数、局部变量等;堆区则用于动态分配内存,由程序员控制释放,共同支撑着程序运行时的数据存储与处理需求。
150 21
|
3月前
|
Windows Python
python获取windows机子上运行的程序名称
python获取windows机子上运行的程序名称
|
3月前
|
小程序 Windows
MASM32编写的程序在Windows 7,10下运行正常,但在Win XP下运行时只闻其声不见其形的故障
MASM32编写的程序在Windows 7,10下运行正常,但在Win XP下运行时只闻其声不见其形的故障
|
3月前
|
C++
【C++基础】程序流程结构详解
这篇文章详细介绍了C++中程序流程的三种基本结构:顺序结构、选择结构和循环结构,包括if语句、三目运算符、switch语句、while循环、do…while循环、for循环以及跳转语句break、continue和goto的使用和示例。
65 2
|
2月前
|
API Windows
Windows之窗口原理
这篇文章主要介绍了Windows窗口原理和如何使用Windows API创建和管理窗口。
65 0
|
2月前
|
安全 API C#
C# 如何让程序后台进程不被Windows任务管理器强制结束
C# 如何让程序后台进程不被Windows任务管理器强制结束
71 0
|
3月前
|
安全 网络安全 API
基于WMI更新Windows系统信息采集程序sysInfo的一些收获
基于WMI更新Windows系统信息采集程序sysInfo的一些收获
|
4月前
|
JavaScript Windows
electron程序运行在某些 windows 上白屏
electron程序运行在某些 windows 上白屏
|
4月前
|
PHP C++ Python
右手坐标系,空间点绕轴旋转公式&程序(Python和C++程序)
右手坐标系,空间点绕轴旋转公式&程序(Python和C++程序)
87 0