操作系统函数允许程序员对控制台屏幕上的输出进行控制。
C++ 编译器提供了用于调用操作系统函数的特殊库,现在来看一下 Windows 操作系统有关屏幕光标定位的函数,此函数是 SetConsoleCursorPosition。
到目前为止,我们创建的所有程序都是在屏幕的顶行显示输出。然后,当用户按回车键或程序输出 endl 或“\n”时,它们一次移动一行。但是,如果正在屏幕的第 5 行写入内容,想要回到第 2 行该怎么办?或者,如果想在屏幕的正中间显示某些东西该怎么办?要满足这些要求,可以在 Windows 系统上使用 SetConsoleCursorPosition 函数,在写入输出之前将光标移动到所需位置。
要使用此函数,必须做以下几件事情:
典型的文本屏幕具有 25 行,每行具有 80 个打印位置。这些位置中的每一个称为格。格是一个可以显示单个字符的小块,它可以通过行号及其在该行上的位置进行区别。行的范围是 0〜24,其中 0 是屏幕的顶行。每行上的打印位置通常被称为列,范围为 0〜79,其中最左侧为 0。格的行和列可以标识它在屏幕上的位置,它们称为其坐标。
要将光标放在指定的屏幕格中,必须通过在 Windows 中定义的 COORD 结构中设置两个变量来指定格坐标。这个结构有两个名为 X 和 Y 的成员变量,X 保存列位置,Y 保存行位置。以下是该结构大致的样子:
struct COORD
{
short int X; // 列位置
short int Y; // 行位置
};
现在来介绍其使用方法。以下代码段可以将单词“Hello”写入标准输出屏幕的中心:
HANDLE screen = GetStdHandle(STD_OUTPUT_HANDLE);
COORD position; // position 是一个 COORD 结构
position.X = 38; //设置列在屏幕中心附近
position.Y = 11; //设置行在屏幕中心附近
//将光标放在那里,然后开始打印
SetConsoleCursorPosition(screen, position);
cout << "Hello" << endl;
注意,在设置一个屏幕位置时,必须在程序的所有输出后面都跟上 endl,这是确保输出实际显示在此位置的必要条件。如果不使用 endl,则在光标位置更改后,输出可能会缓冲并在很久以后写入屏幕。如果在输出后面接着使用换行符“\n”则不起作用,因为它不会像 endl 那样冲刷屏幕缓冲区。
下面的程序定位光标以在屏幕中心附近显示一组嵌套框:
#include <iostream>
#include <windows.h>
using namespace std;
void placeCursor(HANDLE, int, int);
void printStars (int);
int main()
{
const int midRow = 12,midCol = 40, numBoxes = 3;
int width, startRow, endRow;
//Get the handle to standard output device (the console)
HANDLE screen = GetStdHandle(STD_OUTPUT_HANDLE);
// Each loop prints one box
for (int box = 1, height = 1; box <= numBoxes; box++, height+=2)
{
startRow = midRow - box;
endRow = midRow + box;
width = box*5 + (box+1)%2;// Adds 1 if box*5 is an even number
//Draw box top
placeCursor(screen, startRow, midCol-width/2);
printStars(width);
// Print box sides
for (int sideRow = 1; sideRow <= height; sideRow++)
{
placeCursor(screen, startRow + sideRow, midCol-width/2);
cout << '*' << endl;
placeCursor(screen, startRow + sideRow,midCol+width/2);
cout << '*' << endl;
}
// Draw box bottom
placeCursor(screen, endRow, midCol-width/2);
printStars(width);
Sleep(750); //Pause 3/4 second between boxes displayed
}
placeCursor(screen, 20, 0); // Move cursor out of the way return 0;
}
void placeCursor(HANDLE screen, int row, int col)
{
COORD position;
position.Y = row;
position.X = col;
SetConsoleCursorPosition(screen, position);
}
void printStars(int numStars)
{
for (int star = 1; star <= numStars; star++)
cout << '*';
cout << endl;
}
程序输出结果:
*************** * *********** * * * ***** * * * * * * * * * * ***** * * * *********** * ***************
请注意,此程序中使用了 Sleep 函数,它可以使程序执行暂停一段时间,所以它不会使事情发生得太快导致用户看不到它们。传递给函数的实参告诉它应该暂停多少毫秒。毫秒是千分之一秒。因此,假如要暂停执行程序半秒钟,则可以调用以下函数语句:
上面程序运行起来很有趣,但下面程序演示了将光标定位在屏幕上的更实际的应用。
当需要用户输入一系列条目时,不必每次提示一个输入一个,而是可以设计一个屏幕输入表单。这种更专业的从用户获得输入的方式,需要创建和显示一个界面,一次性显示所有提示,然后将光标放置在需要用户输入的特定提示旁边。当用户输入此提示的数据并按下回车键时,光标移动到下一个提示。
#include <iostream>
#include <windows.h> // Needed to set cursor positions
#include <string>
using namespace std;
struct userInfo
{
string name;
int age;
char gender;
};
void placeCursor(HANDLE, int, int); // Function prototypes
void displayPrompts(HANDLE);
void getUserInput (HANDLE, userInfo&);
void displayData (HANDLE, userInfo);
int main()
{
userInfo input;
HANDLE screen = GetStdHandle(STD_OUTPUT_HANDLE);
displayPrompts(screen);
getUserInput(screen, input);
displayData (screen, input);
return 0;
}
void placeCursor(HANDLE screen, int row, int col)
{
COORD position; // holds a pair of X and Y coordinates
position.Y = row;
position.X = col;
SetConsoleCursorPosition(screen, position);
}
void displayPrompts(HANDLE screen)
{
placeCursor(screen, 3, 25);
cout << "******* Data Entry Form *******" << endl;
placeCursor(screen, 5, 25);
cout << "Name: " << endl;
placeCursor(screen, 7, 25);
cout <<"Age: Gender (M/F) : " << endl;
}
void getUserInput(HANDLE screen, userInfo &input)
{
placeCursor(screen, 5, 31);
getline(cin, input.name);
placeCursor(screen, 7, 30);
cin >> input.age;
placeCursor(screen, 7, 55);
cin >> input.gender;
}
void displayData(HANDLE screen, userInfo input)
{
placeCursor(screen, 10,0);
cout<<"Here is the data you entered. \n";
cout<<"Name : "<< input.name << endl;
cout<<"Age : "<< input.name << endl;
cout<<"Gender : "<< input.name << endl;
}
程序输出结果: