鱼C论坛

 找回密码
 立即注册
查看: 1018|回复: 5

[已解决]帮我看看为什么会陷入死循环

[复制链接]
发表于 2021-1-25 14:10:42 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
这几天参照浙江大学翁凯老师的教程学习,写了下面这段程序:
#include<stdio.h>
int main(){
        char a[12][18] = {
                "January Jan." ,
                "February Feb." ,
                "March Mar." ,
                "April Apr." ,
                "May May." ,
                "June Jun." ,
                "July Jul." ,
                "August Aug." ,
                "September Sept." ,
                "October Oct." ,
                "November Nov." ,
                "December Dec." ,               
        };
        int x;
        printf("输入1-12的月份,查找对应的英语,输入“-1”退出:\n");
        scanf("%d", &x);
        while(x!=-1)
        {
                if (x>=1 && x <=12) //过滤掉错误的输入
                {
                        printf("%d月对应的英语是:%s\n", x, a[x-1]);
                }else{
                        printf("输入错误,请输入1-12之间的数,输入“-1”退出:\n");
                }
                scanf("%d", &x);
        }
        return 0;
}

输入整数的时候都能很好的工作,但字母的时候程序则陷入死循环,需要强制退出,想知道这是怎么发生的,要怎么改进。


运行结果如下:

D:\LearnC\20210121>a
输入1-12的月份,查找对应的英语,输入“-1”退出:
2
2月对应的英语是:February Feb.
4
4月对应的英语是:April Apr.
5
5月对应的英语是:May May.
1
1月对应的英语是:January Jan.
-1

D:\LearnC\20210121>a
输入1-12的月份,查找对应的英语,输入“-1”退出:
2
2月对应的英语是:February Feb.
a
2月对应的英语是:February Feb.
2月对应的英语是:February Feb.
2月对应的英语是:February Feb.
2月对应的英语是:February Feb.
2月对应的英语是:February Feb.
2月对应的英语是:February Feb.
......
最佳答案
2021-1-25 15:11:47
二楼说的不对!
这个问题是你对scanf理解不充分导致的。
scanf读取格式化内容,若遇到解析失败,不会清除掉缓冲区内对应的内容
在你这里就是:若输入了字母,那么scanf读取整型失败,但是不会从缓冲区清除你输入的字母,那么下一次循环的时候scanf读到的仍然是那个字母,这就导致了死循环。
解决方法:scanf之后清除缓冲区
        int x=0;
        char ch;
        printf("输入1-12的月份,查找对应的英语,输入“-1”退出:\n");
        scanf("%d", &x);
        while ((ch = getchar()) != EOF && ch != '\n');
        while(x!=-1)
        {
                if (x>=1 && x <=12) //过滤掉错误的输入
                {
                        printf("%d月对应的英语是:%s\n", x, a[x-1]);
                }else{
                        printf("输入错误,请输入1-12之间的数,输入“-1”退出:\n");
                }
                scanf("%d", &x);
                while ((ch = getchar()) != EOF && ch != '\n');
        }
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2021-1-25 14:37:35 | 显示全部楼层
加个getchar(),过滤掉回车就好了
#include<stdio.h>
int main()
{
    char a[12][18] = {
        "January Jan." ,
        "February Feb." ,
        "March Mar." ,
        "April Apr." ,
        "May May." ,
        "June Jun." ,
        "July Jul." ,
        "August Aug." ,
        "September Sept." ,
        "October Oct." ,
        "November Nov." ,
        "December Dec." ,               
        };
    int x;
    printf("输入1-12的月份,查找对应的英语,输入“-1”退出:\n");
    scanf("%d", &x);
    while(x!=-1)
    {
        if (x>=1 && x <=12) //过滤掉错误的输入
        {
            printf("%d月对应的英语是:%s\n", x, a[x-1]);
        }
                else
                {
            printf("输入错误,请输入1-12之间的数,输入“-1”退出:\n");
        }
        getchar();
        scanf("%d", &x);
    }
    return 0;
}
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2021-1-25 15:11:47 | 显示全部楼层    本楼为最佳答案   
二楼说的不对!
这个问题是你对scanf理解不充分导致的。
scanf读取格式化内容,若遇到解析失败,不会清除掉缓冲区内对应的内容
在你这里就是:若输入了字母,那么scanf读取整型失败,但是不会从缓冲区清除你输入的字母,那么下一次循环的时候scanf读到的仍然是那个字母,这就导致了死循环。
解决方法:scanf之后清除缓冲区
        int x=0;
        char ch;
        printf("输入1-12的月份,查找对应的英语,输入“-1”退出:\n");
        scanf("%d", &x);
        while ((ch = getchar()) != EOF && ch != '\n');
        while(x!=-1)
        {
                if (x>=1 && x <=12) //过滤掉错误的输入
                {
                        printf("%d月对应的英语是:%s\n", x, a[x-1]);
                }else{
                        printf("输入错误,请输入1-12之间的数,输入“-1”退出:\n");
                }
                scanf("%d", &x);
                while ((ch = getchar()) != EOF && ch != '\n');
        }
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2021-1-25 15:57:00 | 显示全部楼层
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2021-1-25 19:59:20 | 显示全部楼层
qiuyouzhi 发表于 2021-1-25 14:37
加个getchar(),过滤掉回车就好了

感谢解答,依照您的建议修改后确实不会陷入死循环了,但不能正确的输出“输入错误,请输入1-12之间的数,输入“-1”退出”。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2021-1-25 20:16:29 | 显示全部楼层
sunrise085 发表于 2021-1-25 15:11
二楼说的不对!
这个问题是你对scanf理解不充分导致的。
scanf读取格式化内容,若遇到解析失败,不会清除 ...

感谢解答,您的思路是对的,但还是不能输出错误提示内容,运行结果如下:
D:\LearnC\20210121>a
输入1-12的月份,查找对应的英语,输入“-1”退出:
2
2月对应的英语是:February Feb.
t
2月对应的英语是:February Feb.


最终的解决方法是在scanf()前添加一条重置X=0,代码如下:

        int x;
        char ch;
        printf("输入1-12的月份,查找对应的英语,输入“-1”退出:\n");
        scanf("%d", &x);
        while(x!=-1)
        {
                if (x>=1 && x <=12)
                {
                        printf("%d月对应的英语是:%s\n", x, a[x-1]);
                }else{
                        printf("输入错误,请输入1-12之间的数,输入“-1”退出:\n");
                }
                while ((ch = getchar()) != EOF && ch != '\n');
                x = 0;
                scanf("%d", &x);

        }
        return 0;
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 1 反对 0

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-11-15 08:34

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表