输入缓冲区中的 \n 是我们按下的回车键产生的。getchar从输入缓冲区先读取a,然后putchar打印a,a后面的 \n 会被while循环的getchar再读取,然后putchar进行打印,这样下一次getchar读取的B就实现了换行,循环如此。
此原理可以实现清空缓冲区
举例:
int main() { printf("请输入密码:>"); char password[20] = { 0 }; scanf("%s", &password); printf("请确认密码(Y/N):>"); int ch = getchar(); if (ch == 'Y') { printf("确认成功\n"); } else { printf("确认失败\n"); } return 0; }
运行结果:
Q:为什么会失败呢?
A:这里又运用到了缓冲区的知识,在输入了123456时按下了回车键(\n),scanf先从缓冲区中读走123456,getchar紧接着就读走了 \n ,if判断不是Y,就走了else语句,打印出了”确认失败“字样。
简单修改后代码如下:
int main() { printf("请输入密码:>"); char password[20] = { 0 }; scanf("%s", &password); getchar();//处理\n printf("请确认密码(Y/N):>"); int ch = getchar(); if (ch == 'Y') { printf("确认成功\n"); } else { printf("确认失败\n"); } return 0; }
运行结果:运行结果:
简单的加一句getchar就可以了吗?不是的,如果我们改变输入的内容再试试
Q:为什么又失败了呢?
Q:为什么又失败了呢?
A:scanf 函数在读取的时候只是读走了123,处理 \n 的getchar将a前面的空格处理了,后面的getchar又读取了a,if判断不是Y,就走了else语句,打印出了”确认失败“字样。
那怎么才能做到真正实现正确的程序呢?
这次的处理 \n 的写法:
while (getchar() != '\n') { ; }
用while循环实现读不到 \n 继续往后面读,实现读取 \n 。
最终修改版,代码如下:
int main() { printf("请输入密码:>"); char password[20] = { 0 }; scanf("%s", &password); //清理缓冲区的\n while (getchar() != '\n') { ; } printf("请确认密码(Y/N):>"); int ch = getchar(); if (ch == 'Y') { printf("确认成功\n"); } else { printf("确认失败\n"); } return 0; }
运行结果:
Q:读代码,给出结果
int main() { char ch = '\0'; while ((ch = getchar()) != EOF) { if (ch < '0' || ch > '9') continue; putchar(ch); } return 0; }
运行结果:
这段代码只是读取字符0~9的。
2、for循环
for循环语法
for(表达式1; 表达式2; 表达式3)
循环语句;
//表达式1为初始化部分,用于初始化循环变量的。
//表达式2为条件判断部分,用于判断循环时候终止。
//表达式3为调整部分,用于循环条件的调整。
接下来我们以例子来进一步了解for循环:
Q:使用for循环来打印1~10
int main() { int i = 0; for (i = 1; i <= 10; i++)//i=1为初始化部分;i<=10为判断部分;i++为调整部分。 { printf("%d ", i); } return 0; }
运行结果:
我们将for循环与while循环进行对比:
int main() { int i = 1;//初始化 while (i <= 10)//判断 { printf("%d ", i); i++;//调整 } return 0; }
while循环与for循环写出来的是一样的,但是while循环的初始化,判断,调整三部分比较分散,容易出错。,但是for循环这三部分集中写着,更加整洁方便,且for循环的使用频率比较高。
while循环里有break和continue那么for循环也是有的,接着我们继续了解break和continue在while循环中的使用。
2.1 break在for循环中的使用
Q:看这段代码,它的执行结果是什么呢?
int main() { int i = 0; for (i = 1; i <= 10; i++) { if (5 == i) break; printf("%d ", i); } return 0; }
运行结果:
我们看到运行结果是:1 2 3 4。与while循环的结果是一样的。
这是因为当循环不断进行,调整部分将 i 调整到 i = 5,经过 if 的判断为真后触发了break的运行,break直接跳出了循环,来到了return 0 ;,代码结束。
因此break在for循环的作用:
break在for循环中是终止循环,直接将循环结束掉,在for循环和while循环里是一样的作用。
2.2 continue在for循环中的使用
Q:看这段代码,它的执行结果是什么呢?
int main() { int i = 0; for (i = 1; i <= 10; i++) { if (5 == i) continue; printf("%d ", i); } return 0; }
运行结果:
这里的运行结果是:1 2 3 4 6 7 8 9 10
因为当循环不断进行,将 i 调整到 i = 5 之后,经过 if 的判断为真后,触发了 continue 后跳过了本次循环continue 的后面部分(printf 部分),所以没有打印 5 ,但是它会回到 for 循环的条件语句部分,i++ 正常进行,判断部分正常进行,打印 6 ,循环下去打印 7 8 9 10。不像while循环中的打印1~4 后死循环。
continue在for循环中的作用也是跳过本次循环后面的部分,但在while循环和for循环中有点差异,主要原因是两种循环的调整放在不同的地方。
for循环的建议:
1.不可在for循环体内修改循环变量,防止 for 循环失去控制(循环体内在加上一个调整部分就会出错)。
2.建议for语句的循环控制变量的取值采用“前闭后开区间”的写法(原因是前闭后开可以提高可读性,比如打印1~10)。
2.3 for循环的变种
1. 在for循环中初始化,判断,调整部分都可以省略,这样就会构成死循环。
这三个条件中最重要的是判断部分,如果不写判断部分就可以构成死循环,这是因为省略掉判断部分就意味着判断恒为真,因此就会变成死循环。
int main() { for (; ;) { printf("hehe\n"); } return 0; }
建议:不要轻易省略,这样会导致意外的发生。
Q:读这段代码,给出它的结果
int main() { int i = 0; int j = 0; for (; i < 3; i++) { for (; j < 3; j++) { printf("hehe\n"); } } return 0; }
运行结果:
Q:不应该是9个hehe么,为什么是3个呢?
A:因为 i 作为外循环在第一次循环时,内层循环的 j 在最后的一次调整中 j = 2 ,内层循环结束,来到外层循环的 i++ 调整部分,此时 i = 1 且 i < 3 ,进入到内层循环,这时的 j 没有被初始化为 0 ,而是保留了上一次的 j = 2,经过判断部分不会进入循环,所以打印了 3 次 hehe 。
如何给内层循环加上 j = 0 初始化部分就可以实现 9 个hehe 。
int main() { int i = 0; int j = 0; for (; i < 3; i++) { for (j = 0; j < 3; j++) { printf("hehe\n"); } } return 0; }
运行结果:
2. 循环控制变量可以是多个
举例:
Q:读这段代码,给出它的结果
int main() { int i = 0; int j = 0; for (i = 0, j = 0; i < 3 && j < 5; i++, j++) { printf("hehe\n"); } return 0; }
运行结果: