C语言输入缓冲区实例详解:如何从键盘读取指定的字符串长度

15 篇文章 2 订阅
博客详细讨论了在用户登录过程中,如何安全地从键盘读取用户名并存储到缓冲区,确保不超过预设长度,并处理回车字符。通过示例代码展示了如何避免输入超出限制,以及处理输入缓冲区的内容,确保程序的稳定性。强调了输入检查和字符串终止符的重要性。
摘要由CSDN通过智能技术生成

昨天接触了一下用户登录,发现从键盘读取数据到缓冲区,再保存到程序变量这个过程,还是需要格外注意,因为一不小心,就容易读取错误。下面用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掉了。密码的处理虽然不同,也是同理,大家可以动手试一下

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值