昨天接触了一下用户登录,发现从键盘读取数据到缓冲区,再保存到程序变量这个过程,还是需要格外注意,因为一不小心,就容易读取错误。下面用getchar函数说明一下:
getchar()函数
函数原型: int getchar(void)
函数功能: 从标准输入 stdin 获取一个字符
函数返回值: 该函数以无符号 char 强制转换为 int 的形式返回读取的字符
使用场景: 用户登录时,输入用户名和密码,程序将数据保存到变量,用来进行显示,验证等其他操作
下面是用户结构体:
//用户结构体,用来发送给服务器
struct account{
char user[20];
char pw[7];
};
typedef struct _PACKET_ACCOUNT{
struct account acc;
int result;
char errmsg[128];
}PACKET_ACCOUNT;
当然,如果只是测试,后面的PACKET_ACCOUNT可不用。首先说用户名读取。
数组的最后一个元素为终止字符串'\0',所以一共读取不超过19个字符作为用户名,那么
PACKET_ACCOUNT pack;//pack.acc.user[index]表示数组元素
同时满足以下三个条件:
1 读取不超过19个字符即为成功
2 可以为空,为空时设置缺省用户名
3 超过19个字符重新输入
这时候就要解决几个问题比如,如果用户输入的长度超过19怎么办?正好等于19怎么办?因为字符串的最后一个字符是终止符(编译器自动添加,如果用字符数组保存字符串,必须腾出最后一个元素给终止符),还要考虑正好输入19个字符按下回车后,第20个字符的处理。
输入缓冲区就像中午食堂吃饭排队的队伍。如果前面的人不走,后面的没法打饭。如果有10个人排队,你想把红烧肉给最后面的女朋友,那你是不是得先把前面9个安排了(读取9个字符)。然而,如果用%s输出字符串,这样是不安全的,要保证第十个字符是‘\0’。
char buf[10];
int i = 0;
while((buf[i] = getchar())){//循环读取输入缓冲区字符,一次读一个
if(i == 9){ //读到第10个
buf[i] = '\0'; //把从键盘读到的第十个字符重新赋值为'\0'
break;
i++;
}
所以,当我们的用户名数组为buf[20]时,要保证最后一个字符是终止符。即按下回车后,(如果输入不大于19个字符按回车)用户名数组里的内容是 '用户名'+ '\n',数组最后一个字符刚好是回车符\n,把它替换成'\0'就可以表示字符串了。buf[strlen(buf)] = ''\0;
PACKET_ACCOUNT pack;
//用户名输入
//cout << "请输入用户名:" << endl;
printf("请输入用户名:\n");
//以下两句作用是清空输入缓冲区,作用是清空输入缓冲区,直到\n为止
while ((c = getchar()) != '\n');
//cin.ignore((std::numeric_limits< streamsize >::max)(), '\n');
int pos = 0;
int size = sizeof(pack.acc.user)/sizeof(char);
while((pack.acc.user[pos] = getchar()))
{
//用户未输入时,设置缺省名称
if(pack.acc.user[0] == '\n'){
sprintf(pack.acc.user,"defalt_user_name");
break;
}
//用户名超过指定长度,重新输入
if(pos > size - 1){
//cout << "用户名太长,请重新输入" << endl;
while ((c = getchar()) != '\n');//输入缓冲区有多余字符,清空掉
memset(pack.acc.user,0,size);
pos = 0;
continue;
}
//读取<=size-1个的字符时,设置读取结束条件
if(pos <= size - 1 && pack.acc.user[pos] == '\n')
{
//cout << "读取<=19字符,用户名已成功录入!" << endl;
pack.acc.user[pos] = '\0';
break;
}
pos++;
}
这样,不管你怎么输入,程序就都不会因为缓冲区而down掉了。密码的处理虽然不同,也是同理,大家可以动手试一下