• 关于用scanf(“%d“,)时非法字符输入导致的死循环的原因及解决方法


    现象描述

    题目要求输入一个整数n(0字符输入的情况。若输入非法字符或者不在合法区间,则重新输入。于是我们很自然的打出了如下代码:

    int a;
    while(1){
    	scanf("%d",&a);
    	if(a>0&&a<20) break;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 我们运行一下后就会发现,一旦我们输入了一个非法字符(如a,aaa,asdas),那么我们接下来输什么都没有用,程序陷入了死循环。

    原理分析:

    • 经多方查找资料得知,scanf有缓存区,当输入非法字符(要求的类型与输入的类型不符合),scanf会直接跳过,即该输入不会被接收也不会被清除,而会被存放在scanf的缓存区,当下次调用scanf函数时,会直接从缓存区读取非法字符,造成死循环。

    解决思路:

    • 清除滞留在scanf缓存区的数据,使下一次循环时scanf可以正常的读取用户输入。

    解决方法

    1. 用getchar把缓冲区清空。
    //法一:
    int a;
    while(1){
    	scanf("%d",&a);
    	if(a>0&&a<20) break;
    	while(getchar()!='\n'); //清空缓存区,直到换行输入
    } 
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    //法二:
    int a,m=0;
    do{
    	m=scanf("%d",&a);//m是scanf读入的数的个数 
    	if(m!=1)//如果读入了一个数 即 输入了一个s整数 
    		while(getchar()!='\n'); //清空缓存区,直到换行输入
    }while(m!=1||a<=0||a>=20);
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    1. 直接使用缓冲区清除函数fflush(stdin); //stdin代表标准输入
    int a;
    while(1){
    	fflush(stdin);//在每次输入前清空缓冲区
    	scanf("%d",&a);
    	if(a>0&&a<20) break;
    } 
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
  • 相关阅读:
    JQuery系列之事件
    精通Linux,不清楚Telnet和SSH?
    SpringAOP补充-通知获取类型
    GPS卫星位置解算原理与程序设计(C语言)
    vue路由
    交叉编译 笔记
    机器学习强基计划2-3:图文详解决策树预剪枝、后剪枝原理+Python实现
    七夕玫瑰花合集
    【Java 进阶篇】JDBC PreparedStatement 详解
    树莓派安装我的世界java版
  • 原文地址:https://blog.csdn.net/Robin2321/article/details/132774269