控制台API函数----HANDLE、SetConsoleCursorPosition、SetConsoleTextAttribute

控制台API函数

调用相关文本界面控制的API函数,这些函数可分为三类。

一、用于控制台窗口控制的函数(包括窗口的缓冲区大小、窗口前景字符和背景颜色、窗口标题、大小和位置等);

二、用于控制台输入输出的函数(包括字符属性操作函数);

三、其他的函数并为最后一类。 

(注意:当不需要使用句柄时需要调用CloseHandle()来关闭输入输出句柄,就像malloc申请的内存空间最后需要用free函数释放掉一样,后面会介绍CloseHandle的具体使用方法)

可以理解为句柄就是一种资源,就像内存一样,用完就得释放,一个程序创建的句柄数是有限制的,时间如果比较短可能看不出来什么问题,时间一旦长了,可能就会导致程序无法继续创建句柄(意思是系统的句柄资源用光了)

 

控制台窗口控制API函数(部分)

GetConsoleScreenBufferInfo 获取控制台窗口信息
GetConsoleTitle 获取控制台窗口标题
ScrollConsoleScreenBuffer 在缓冲区中移动数据块(更改指定缓冲区大小)
SetConsoleScreenBufferSize 更改指定缓冲区大小
SetConsoleTitle 设置控制台窗口标题
SetConsoleWindowInfo 设置控制台窗口信息

(函数前半部分很相似SetConsole(设置控制台)、GetConsole(获取控制台),只需要记后面就方便多了)

在程序中还必须包含头文件windows.h

 

1、HANDLE

本篇随笔主要内容是各个函数的使用,想了解句柄HANDLE的读者可以先看下这篇文章:深入了解Windows句柄到底是什么    

HANDLE在WindNT.h中的声明为

typedef void *HANDLE;

从上面可以看出HANDLE是一种无类型指针,句柄是处理对象的一个接口,你可以通过句柄去操作程序中所涉及的对象。在windows中,句柄是和对象一一对应的32位无符号整数值,对象可以映射到唯一的句柄,

句柄也可以映射到唯一的对象windows需要向程序员提供必要地编程接口,在这些接口中,允许程序员访问,创建和销毁对象,但是,出于封装的考虑,windows并不想向程序员返回指针

如果作数据的话,句柄这种方式则允许你按自己的方式直接操作数据,但windows又不向你直接暴露数据。直接操作数据是程序员需要的,不暴露数据是windows所需要的,句柄封装方式实现了各取所需。

(很好的找到了满足双方需且的平衡点)

“句柄”是Windows最常用的概念。它通常用来标识Windows资源(如菜单、图标、窗口等)和设备等对象。虽然可以把句柄理解为是一个指针变量类型,但它不是对象所在的地址指针,而是作为Windows系统内部表的索引值来使用的。 

 代码示例:

HANDLE hOut;

 

2、GetStdHandle函数

函数功能:获取指定的标准设备的句柄,使用GetStdHandle需要一个参数,参数的取值有三种

STD_INPUT_HANDLE----标准输入句柄

STD_OUTPUT_HANDLE----标准输出句柄

STD_ERROR_HANDLE----标准错误句柄

先回顾一下文件操作中输入输出重定向的概念

在默认情况下:

标准输入(stdin)----键盘

标准输出(stdout)----显示器(屏幕)

标准错误(stderr)----显示器(屏幕)

(注意:标准输出句柄和标准错误句柄默认情况下都是对应的屏幕)

代码示例:

HANDLE hOut;
hOut = GetStdHandle(STD_OUTPUT_HANDLE); /* 获取标注输出句柄 */

 HANDLE hOut可理解为:定义了一个变量名为hOut的变量,变量的类型为HANDLE(句柄)类型

 

3、COORD

COORD是Windows API中定义的一种结构,表示一个字符在控制台屏幕上的坐标。其在头文件WinCon.h中的定义为:

1 typedef struct _COORD {
2     SHORT X;
3     SHORT Y;
4 } COORD, *PCOORD;

从上面可以看出 COORD是一个结构体类型,结构体里面有两个短整形变量X、Y,用来表示光标的坐标

(注意:X是横坐标(horizontal coordinate),Y是纵坐标(vertical coordinate)不要与二维数组的坐标表示搞混了,二维数组第一维是纵坐标,第二维是横坐标)

代码示例:

COORD pos = {x, y};/* x是横坐标 y是纵坐标 */

 

4、CONSOLE_CURSOR_INFO

控制台光标信息,先去WinCon.h里面去瞧一瞧声明是什么样子的

typedef struct _CONSOLE_CURSOR_INFO {
    DWORD  dwSize;
    BOOL   bVisible;
} CONSOLE_CURSOR_INFO, *PCONSOLE_CURSOR_INFO;

从上面可以看出,CONSOLE_CURSOR_INFO是一个结构体,结构体里面有两个成员一个是dwSize表示光标的大小(取值范围1~100),第二个是成员是bVisible表示光标是否可见((TRUE)可见,(FALSE)不可见)

 

5、SetConsoleCursorInfo函数

函数功能:设置控制台光标信息,使用这个函数需要两个参数,第一个参数的类型HADNLE,第二个参数的类型CONSOLE_CURSOR_INFO *类型

代码示例:

 1 #include <stdio.h>
 2 #include <windows.h>
 3 
 4 void ConsoleCursor ();
 5 
 6 int main() 
 7 {
 8     
 9 
10     return 0;
11 }
12 
13 void ConsoleCursor ()
14 {
15     HANDLE hOut;
16     CONSOLE_CURSOR_INFO cur;    /* 定义了一个结构体变量 cur*/
17 
18     cur.dwSize = 100;            /*  光标大小设置为100 */
19     hOut = GetStdHandle(STD_OUTPUT_HANDLE);    /* 获取标注输出句柄 */
20     SetConsoleCursorInfo(hOut, &cur);    /* 设置控制台光标信息 */
21 }

上面的代码中,ConsoleCursor这个函数中没有设置光标是否可见(默认为可见),读者如果想设置为光标不可见将bVisible赋值为0就可以了(0(光标不可见)、非0(光标可见))

(注意:dwSize的取值范围是1~100,如果赋值的大小超过100则参数失效,光标大小为默认大小)

运行效果图:

 

在设置光标不可见时发现了一个有趣的现象,如果只把光标设置为不可见cur.bVisible = 0,而不对光标的大小赋值的话,用SetConsoleCursorInfo是不起作用的

 代码如下:

 1 #include <stdio.h>
 2 #include <windows.h>
 3 
 4 void ConsoleCursor ();
 5 
 6 int main() 
 7 {
 8     printf("***");
 9     ConsoleCursor();
10     return 0;
11 }
12 
13 void ConsoleCursor ()
14 {
15     HANDLE hOut;
16     CONSOLE_CURSOR_INFO cur;    /* 定义了一个结构体变量 cur*/
17 
18     //cur.dwSize = 100;    /*  光标大小设置为100 */
19     cur.bVisible = 0;    /* 光标设置为不可见 */
20     hOut = GetStdHandle(STD_OUTPUT_HANDLE);    /* 获取标注输出句柄 */
21     SetConsoleCursorInfo(hOut, &cur);    /* 设置控制台光标信息 */
22 }

运行效果如下:

光标并没有隐藏,想要隐藏光标需要给光标大小cur.dwSize赋值,赋值范围1~100(如果赋值大小超过1~100,那么赋值无效,光标还是无法隐藏---这也算个坑了 要注意一下)

给cur赋初值也可这样写

CONSOLE_CURSOR_INFO cur = {1, 0};

 

 

 

 

6、SetConsoleCursorPosition函数

SetConsoleCursorPosition这个函数的功能是:设置控制台光标坐标(Set--设置、Console--控制台、Cursor--光标、Position--坐标,还算蛮好记的),使用这个函数需要两个参数:第一个参数类型为HANDLE,第二个参数类型为COORD

代码示例:

 1 #include <stdio.h>
 2 #include <windows.h>
 3 
 4 void gotoxy(int x, int y);
 5 void ConsoleCursor ();
 6 
 7 int main() 
 8 {
 9     printf("***");
10     gotoxy(0,5);
11     printf("***");
12 
13     return 0;
14 }
15 
16 void gotoxy(int x, int y)
17 {
18     COORD pos = {x, y};/* x是横坐标 y是纵坐标 */
19     HANDLE hOut;
20 
21     hOut = GetStdHandle(STD_OUTPUT_HANDLE); /* 获取标注输出句柄 */
22     SetConsoleCursorPosition(hOut,pos);        /* 设置控制台光标坐标(设备句柄, 光标坐标) */
23 }

 运行效果图:

 

7、CONSOLE_SCREEN_BUFFER_INFO

控制台窗口缓冲区信息,先到WinCon.h头文件中瞧一下声明部分

typedef struct _CONSOLE_SCREEN_BUFFER_INFO {
    COORD dwSize;
    COORD dwCursorPosition;
    WORD  wAttributes;
    SMALL_RECT srWindow;
    COORD dwMaximumWindowSize;
} CONSOLE_SCREEN_BUFFER_INFO, *PCONSOLE_SCREEN_BUFFER_INFO;

这个比上面CONSOLE_CURSOR_INFO(控制台光标信息)的成员多了一些,一个一个成员来看

第一个成员:

COORD dwSize; 

dwSize缓冲区的宽度X和高度Y

 

第二个成员:

COORD dwCursorPosition;

dwCursorPosition当前光标在缓冲区中的坐标

 

第三个成员:

WORD  wAttributes;

wAttributes文本的属性(前景色、背景色等信息)

 

第四个成员:

SMALL_RECT srWindow;

srWindow当前窗口显示的大小和位置

 

第五个成员:

COORD dwMaximumWindowSize;

dwMaximumWindowSize最大的窗口缓冲区大小

 

8、GetConsoleScreenBufferInfo函数

函数功能:获取控制台窗口缓冲区信息,使用这个函数需要两个参数,第一个参数HANDLE类型,第二参数CONSOLE_SCREEN_BUFFER_INFO*类型

使用这个函数就可以得到控制台缓冲区的一些有意思的信息了,下面来看一下可以得到哪些参数

代码示例:

#include <stdio.h>
#include <windows.h>

void TextAttribute();        /* 设置前景色、后景色、获取控制台信息 */

int main() 
{
    printf("***");
    TextAttribute();
    return 0;
}
/* Function: 设置前景色后景色和获取控制台信息 */
void TextAttribute ()
{
    HANDLE hOut;
    CONSOLE_SCREEN_BUFFER_INFO scr;

    hOut = GetStdHandle(STD_OUTPUT_HANDLE);    /* 获取标准输出句柄 */
    GetConsoleScreenBufferInfo(hOut, &scr);
    printf("当前光标:\n横坐标=%d  纵坐标=%d\n", scr.dwCursorPosition.X, scr.dwCursorPosition.Y);
    printf("缓冲区最大:宽=%d,高=%d\n",scr.dwMaximumWindowSize.X, scr.dwMaximumWindowSize.Y);
    printf("缓冲区 宽=%d  高=%d\n",scr.dwSize.X, scr.dwSize.Y);
    printf("底部=%d 顶部=%d 左=%d 右=%d", scr.srWindow.Bottom, scr.srWindow.Top, scr.srWindow.Left, scr.srWindow.Right);
    printf("",scr.wAttributes);
    SetConsoleTextAttribute(hOut, FOREGROUND_BLUE); /* 设置控制台颜色,前景色为蓝色*/
    SetConsoleTextAttribute(hOut, BACKGROUND_RED);    /* 背景色为红色 */
}

 运行结果:

关于上面这些参数可以在控制台的默认值里面找到对应的值底部和右的值需要在窗口最大的宽度下-1,这里跟数组的下标有点相似,窗口大小也是从0开始的

 

9、SetConsoleTextAttribute函数

 SetConsoleTextAttribute这个函数的功能是:设置控制台文本属性(颜色),可以设置前景色FOREGROUND(文本颜色)和背景色BACKGROUND(看厌了DOS控制台的黑白色,终于可以换个颜色了(┬_┬))

使用SetConsoleTextAttribute需要两个参数(句柄, 属性),第二个参数没写成颜色是因为SetConsoleTextAttribute还可以高亮显示,现在我们去WinCon.h头文件里面去瞧一瞧,看看第二个参数有哪些是可以用的

#define FOREGROUND_BLUE      0x0001 // text color contains blue.
#define FOREGROUND_GREEN     0x0002 // text color contains green.
#define FOREGROUND_RED       0x0004 // text color contains red.
#define FOREGROUND_INTENSITY 0x0008 // text color is intensified.
#define BACKGROUND_BLUE      0x0010 // background color contains blue.
#define BACKGROUND_GREEN     0x0020 // background color contains green.
#define BACKGROUND_RED       0x0040 // background color contains red.
#define BACKGROUND_INTENSITY 0x0080 // background color is intensified.
#define COMMON_LVB_LEADING_BYTE    0x0100 // Leading Byte of DBCS
#define COMMON_LVB_TRAILING_BYTE   0x0200 // Trailing Byte of DBCS
#define COMMON_LVB_GRID_HORIZONTAL 0x0400 // DBCS: Grid attribute: top horizontal.
#define COMMON_LVB_GRID_LVERTICAL  0x0800 // DBCS: Grid attribute: left vertical.
#define COMMON_LVB_GRID_RVERTICAL  0x1000 // DBCS: Grid attribute: right vertical.
#define COMMON_LVB_REVERSE_VIDEO   0x4000 // DBCS: Reverse fore/back ground attribute.
#define COMMON_LVB_UNDERSCORE      0x8000 // DBCS: Underscore.

 从上面的代码中可以看出前景色FOREGROUND和后景色BACKGROUND都只提供了三原色(红RED、绿GREEN、蓝BLUE)和颜色加强,后面的7个参数这里不再详述

代码示例:

 1 #include <stdio.h>
 2 #include <windows.h>
 3 
 4 void gotoxy(int x, int y);
 5 void ConsoleCursor ();
 6 void TextAttribute();
 7 
 8 int main() 
 9 {
10     printf("***\n");
11     TextAttribute();
12     printf("***");
13 
14     return 0;
15 }
16 
17 void TextAttribute ()
18 {
19     HANDLE hOut;
20 
21     hOut = GetStdHandle(STD_OUTPUT_HANDLE);    /* 获取标注输出句柄 */
22     SetConsoleTextAttribute(hOut, FOREGROUND_BLUE); /* 设置控制台颜色,前景色为蓝色*/
23     SetConsoleTextAttribute(hOut, BACKGROUND_RED);    /* 背景色为红色 */
24 }

运行效果

也可在SetConsoleTextAttribute中同时设置前景色和后景色,代码如下,效果和上图相同

SetConsoleTextAttribute(hOut, FOREGROUND_BLUE|BACKGROUND_RED);    /* 设置控制台颜色,前景色为蓝色,背景色为红色*/

 10、GetConsoleTitle函数和GetConsoleTitleA函数

GetConsoleTitle函数功能是:获取控制台标题,使用这个函数需要两个参数,第一个是LTSPR类型,第二个是DWORD类型

GetConsoleTitleA函数功能和上面相同,使用时参数有点区别,第一个是LPSTR类型,第二个是DWORD类型

代码示例:

 

 1 #include <stdio.h>
 2 #include <windows.h>
 3 
 4 void ConsoleTitle();        /* 获取控制台标题 */
 5 int main() 
 6 {
 7     ConsoleTitle();
 8     return 0;
 9 }
10 
11 /* Function: 获取控制台标题 */
12 void ConsoleTitle()
13 {
14     char Title[200];
15 
16     SetConsoleTitleA("My title");
17     GetConsoleTitleA(Title,200);
18     puts(Title);
19 }

 

 

 

运行效果:

 

11、CloseHandle函数

这个函数和fclose很相似这里就不再赘述了

posted @ 2019-03-15 21:35  Luv3  阅读(11521)  评论(1编辑  收藏  举报