鱼C论坛

 找回密码
 立即注册
查看: 1159|回复: 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之后清除缓冲区

  1.         int x=0;
  2.         char ch;
  3.         printf("输入1-12的月份,查找对应的英语,输入“-1”退出:\n");
  4.         scanf("%d", &x);
  5.         while ((ch = getchar()) != EOF && ch != '\n');
  6.         while(x!=-1)
  7.         {
  8.                 if (x>=1 && x <=12) //过滤掉错误的输入
  9.                 {
  10.                         printf("%d月对应的英语是:%s\n", x, a[x-1]);
  11.                 }else{
  12.                         printf("输入错误,请输入1-12之间的数,输入“-1”退出:\n");
  13.                 }
  14.                 scanf("%d", &x);
  15.                 while ((ch = getchar()) != EOF && ch != '\n');
  16.         }
复制代码
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2021-1-25 14:37:35 | 显示全部楼层
加个getchar(),过滤掉回车就好了
  1. #include<stdio.h>
  2. int main()
  3. {
  4.     char a[12][18] = {
  5.         "January Jan." ,
  6.         "February Feb." ,
  7.         "March Mar." ,
  8.         "April Apr." ,
  9.         "May May." ,
  10.         "June Jun." ,
  11.         "July Jul." ,
  12.         "August Aug." ,
  13.         "September Sept." ,
  14.         "October Oct." ,
  15.         "November Nov." ,
  16.         "December Dec." ,               
  17.         };
  18.     int x;
  19.     printf("输入1-12的月份,查找对应的英语,输入“-1”退出:\n");
  20.     scanf("%d", &x);
  21.     while(x!=-1)
  22.     {
  23.         if (x>=1 && x <=12) //过滤掉错误的输入
  24.         {
  25.             printf("%d月对应的英语是:%s\n", x, a[x-1]);
  26.         }
  27.                 else
  28.                 {
  29.             printf("输入错误,请输入1-12之间的数,输入“-1”退出:\n");
  30.         }
  31.         getchar();
  32.         scanf("%d", &x);
  33.     }
  34.     return 0;
  35. }
复制代码
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2021-1-25 15:11:47 | 显示全部楼层    本楼为最佳答案   
二楼说的不对!
这个问题是你对scanf理解不充分导致的。
scanf读取格式化内容,若遇到解析失败,不会清除掉缓冲区内对应的内容
在你这里就是:若输入了字母,那么scanf读取整型失败,但是不会从缓冲区清除你输入的字母,那么下一次循环的时候scanf读到的仍然是那个字母,这就导致了死循环。
解决方法:scanf之后清除缓冲区

  1.         int x=0;
  2.         char ch;
  3.         printf("输入1-12的月份,查找对应的英语,输入“-1”退出:\n");
  4.         scanf("%d", &x);
  5.         while ((ch = getchar()) != EOF && ch != '\n');
  6.         while(x!=-1)
  7.         {
  8.                 if (x>=1 && x <=12) //过滤掉错误的输入
  9.                 {
  10.                         printf("%d月对应的英语是:%s\n", x, a[x-1]);
  11.                 }else{
  12.                         printf("输入错误,请输入1-12之间的数,输入“-1”退出:\n");
  13.                 }
  14.                 scanf("%d", &x);
  15.                 while ((ch = getchar()) != EOF && ch != '\n');
  16.         }
复制代码
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2021-1-25 15:57:00 | 显示全部楼层
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

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

感谢解答,依照您的建议修改后确实不会陷入死循环了,但不能正确的输出“输入错误,请输入1-12之间的数,输入“-1”退出”。
小甲鱼最新课程 -> https://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;
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 1 反对 0

使用道具 举报

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

本版积分规则

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

GMT+8, 2025-5-2 02:25

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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