初阶C语言:分支与循环(3)--拓展篇

简介: C语言中的分支与循环语句分步详解,承接上篇的循环语句,本篇是关于分支与循环语句的拓展,包含内存缓存区、关机小程序、二分查找、有趣的小练习。

在学习完分支与循环之后,相信大家对分支语句、循环语句都有了一定的掌握,为了是我们的基础知识更扎实,在此基础上我们在来了解一些新的东西

紫蓝色几何渐变科技互联网微信公众号封面 (1).gif

1.清理输入缓存区

在以前的学习中我们学习过scanf (输入) 和printf (输出) 。其实还有getchar和putchar

getchar---读取一个字符 putchar---输出一个字符

代码演示:

#include <stdio.h>intmain()
{
//由于getchar定义的返回类型就是int类型,所以使用int类型来接收intch=getchar();
putchar(ch);
return0;
}

可以看到输入一个字符‘A’就可以打印出一个字符‘A’

输入缓冲区:

当我们使用一系列的输入输出函数时,在输入的时候,函数并不是直接从电脑上键盘上面读取,在函数和键盘中间还有一个输入缓冲区,每当使用输入函数的时候,输入函数都会在输入缓冲区中读取信息,所以在我们电脑键盘上敲出的字符都会先存放在输入缓冲区,然后让输入函数来读取,使用上面的代码来演示一下:

当我们在电脑键盘上敲出A还有回车(Enter),那么在输入缓冲区中就放进去了A和回车(由于回车可以使输入函数启动,再加上回车还有换行的功能也就是转义字符\n),接下来输入函数就要读取,当读取到A时就打印A。

getchar读取到了字符返回的是该字符的ASCII码值,这也就是为什么需要用整形int类型来接收,如果读取失败,则会返回EOF EOF其实等于-1,因为ASCII码值的范围是0~127,所以呢,如果读取失败就证明读取的不在0~127之内所以就返回-1,用int整形来接收再合适不过了 scanf函数默认读取到空格就结束

代码演示:

#include <stdio.h>intmain()
{
intch=0;
while ((ch=getchar()) !=EOF)  //这里表示如果输入的字符符合条件就可以进入循环    {                                //因此通过这样一种方法就可以实现多组输入putchar(ch);
    }
return0;
}

1.1用户登录系统实现

讲了半天也没有讲到怎么清空输入缓存区,那么接下来就到了清空缓存区的时间了:

我们来实现一个简单版的用户登录密码:在屏幕上输入密码,然后确认密码,判断是否确认成功。

#include <stdio.h>intmain()
{
printf("请输入密码:>");
charpassward[20] = { 0 };
//输入密码scanf("%s", &passward);
//确认密码printf("请确认密码(Y/N):>");
intch=getchar();
//判断是否确认成功if (ch=='Y')
    {
printf("确认成功\n");
    }
else    {
printf("确认失败\n");
    }
return0;
}

通过上述代码,关于密码输入的基本逻辑已经建立好了,我们来运行一下看能不能得到我们想要的效果

可以看到,还没等我们来得及确认就已经确认失败了,这是为什么呢?

当我们输入密码时顺带还敲出了回车用来启动scanf函数用来接收密码,所以呢,密码ABCDEF被scanf函数接收,这时,输入缓冲区里面还剩下了\n,这时getchar函数就进行读取,当getchar读取到了\n,发现ch==\n并不是ch==‘Y’,所以呢直接就会确认失败

因此我们需要再加上一个getchar用来将敲出来的回车(\n),拿出来清空输入缓存区

优化代码:

#include <stdio.h>intmain()
{
printf("请输入密码:>");
//输入密码charpassward[20] = { 0 };
scanf("%s", &passward);
//清理缓存区getchar();    //使用getchar再读取一个字符以便达到清除‘\n’//确认密码printf("请确认密码(Y/N):>");
intch=getchar();
//判断是否确认成功if (ch=='Y')
    {
printf("确认成功\n");
    }
else    {
printf("确认失败\n");
    }
return0;
}

可以看到,使用getchar清理输入缓存区就得以实现了,写到这里这个代码还是有欠缺的,还并不是很完美,假如不小心输入了123456 abc\n,这时还是会报错

我们来简单的分析一下:

scanf默认是读取到空格结束,所以scanf读取了123456,剩下了  abc\n,但是getchar一次只能读取一个字符,所以getchar只读取了空格,还剩下abc\n,在判断的时候ch又不等于Y,因此又会确认失败

所以我们可以使用循环,让getchar读取多次,直到读取到‘\n’

代码重优化:

#include <stdio.h>intmain()
{
printf("请输入密码:>");
//输入密码charpassward[20] = { 0 };
scanf("%s", &passward);
//清理缓存区while (getchar() !='\n')
    {
        ;      //由于我们读取到的密码不需要任何操作所以循环语句可以空着    }
//确认密码printf("请确认密码(Y/N):>");
intch=getchar();
//判断是否确认成功if (ch=='Y')
    {
printf("确认成功\n");
    }
else    {
printf("确认失败\n");
    }
return0;
}

写到这里我们的用户登录系统就简单的介绍完了,也让我们了解了使用getchar来清空输入缓存区。

2.关机小程序

在我们电脑上面有许许多多指令,这些指令都是可以在电脑上面使用的:

shutdown -a 取消关机 shutdown -s 关机 shutdown -f 强行关闭应用程序 shutdown -l 注销当前用户 shutdown -r 关机并重启 shutdown -s -t 时间 设置关机倒计时 shutdown -r -t 时间 设置重新启动倒计时 cls 清空屏幕 当我们在代码中使用Windows命令时需要用到 system 同样也需要包含头文件 <stdlib.h> 使用 strcmp 用来比较两个字符串相不相等,在使用时得包含头文件 <string.h>, 当比较的两个字符相等时会返回0

关机小程序需要用到电脑上面的指令,当程序运行起来60秒之后就会关机,如果输入:我是猪,则会取消关机。

代码演示:

//关机小程序#include <stdio.h>#include <string.h>#include <stdlib.h>intmain()
{
charinput[20] = { 0 };
system("shutdown - s - t 60");    //执行60s关机指令while (1)
    {
printf("您的电脑将会在60秒之后关机,如果您输入:我是猪,则可以取消关机!\n请输入:>");
scanf("%s", input);
if (0==strcmp(input, "我是猪")) //用来判断是否输入正确        {
system("shutdown -a");      //如果输入成功则会取消关机break;                     //如果输入正确就会成功跳出循环,不再执行        }                             //如果输入不正确则会重复输入    }
return0;
}

小编在这里就不演示了,各位老铁有兴趣可以在自己电脑上面试一下。

前面我们了解过go to语句,当需要跳转出多层代码时使用go to语句就很方便,同样我们也可以使用go to语句来实现关机小程序:

代码演示:

#include <stdio.h>#include <string.h>#include <stdlib.h>intmain()
{
charinput[20] = { 0 };
system("shutdown - s - t 60");
again:
printf("您的电脑将会在60秒之后关机,如果您输入:我是猪,则可以取消关机!\n请输入:>");
scanf("%s", input);
if (0==strcmp(input, "我是猪"))
    {
system("shutdown -a");
    }
else//如果输入错误又可以跳转到again的开始重新执行输入    {
gotoagain;     
    }
return0;
}

3.折半查找算法

折半查找算法也叫做二分查找法:简单的理解就是每一次查找比上一次查找的范围缩小一半,这样的算法可以大大提高查找效率,要使用折半查找,首先数组得是有序数组 在一个有序数组中查找某个数 要查找一个数组中的某个数,首先得知道这个数组中的元素个数,以便用来设置循环次数,然后再找到这个数的下标,然后通过下标找到这个数 普通方法:遍历整个数组的下标,然后进行查找,找到了就返回下标,找不到就返回0
//普通方法#include <stdio.h>intFindNum(intarr[], intnum, intsz)
{
inti=0;
for (i=0; i<sz; i++)  //遍历整个数组来寻找这个数    {
if (num==arr[i])
        {
returni;          //如果找到了就返回下标        }
    }
return0;
}
intmain()
{
intarr[10] = { 1,2,3,4,5,6,7,8,9,10 };
intnum=0;
scanf("%d", &num);
intsz=sizeof(arr) /sizeof(arr[0]); //计算数组中的元素个数,然后来控制遍历循环次数intret=FindNum(arr,num,sz);
if (ret==0)
    {
printf("很抱歉,找不到!\n");
    }
else    {
printf("找到了,下标是:%d\n", ret);
    }
return0;
}

但是这种遍历的方法效率不高,因为它进行了许多不必要的步骤,假如这个数组有非常多的元素,需要查找其中的一个元素,如果使用遍历就会导致查找速度下降,效率降低,很有可能无法达到所预期的效果,因此我们需要一种新的算法用来省去其中不必要的一系列步骤,从而达到提高效率的目的。 折半查找算法 在生活中也有很多实例可以说明折半查找算法的重要性:比如你的好哥们买了个好电脑,你如果问他这个电脑多少钱,总不可能从1开始依次询问(这种方法相当于C语言中的遍历),我们是不是先给他说一个数,然后他根据你说的这个数来判断是大了还是小了(这种方法相当于C语言中的折半查找法) 使用折半查找算法来找出一个有序数组中的一个数 首先我们得知道这个数组中的元素个数,然后设置两个变量,一个变量表示最左边的元素的下标、一个元素表示最右边的元素的下标、使用左下标和右下标求出中间下标,然后用这个数跟中间下标的元素进行比较,如果这个数比中间下标大,就证明这个数在中间下标和最右边元素的之间,这时就要将最左边的下标移动到这个中间下标,再使用移动后的下标和右下标再求一个中间下标,再使用这个数和重新求得的数比较,依次类推,直到找到这个数,所以需要将这些比较的过程放进一个循环,这个循环中的循环判断条件是左下标<=右下标,

如果查找到比arr[mid]小,就将right挪到mid-1的位置,然后重复这个过程

代码演示:

//二分查找#include <stdio.h>intFindNum(intarr[], intnum, intsz)
{
intleft=0;
intright=sz-1;
while (left<=right)
    {                                       //如果单纯的写(left+right)/2,数据有可能溢出intmid=left+ (right-left) /2; //为了防止数据溢出if (num>arr[mid])       
        {
//下标交换left=mid+1;
        }
elseif (num<arr[mid])
        {
right=mid-1;
        }
else        {
returnmid;   //如果找到则返回下标        }
    }
return0;
}
intmain()
{
intarr[10] = { 1,2,3,4,5,6,7,8,9,10 };
intnum=0;
scanf("%d", &num);
intsz=sizeof(arr) /sizeof(arr[0]);
intret=FindNum(arr,num,sz);
if (ret==0)
    {
printf("很抱歉,找不到!\n");
    }
else    {
printf("找到了,下标是:%d\n", ret);
    }
return0;
}
:使用折半查找算法的前提是数组是有序数组 要注意求中间下标时防止数据溢出 交换下标时要注意交换是否正确

4.有趣的代码练习

编写代码,演示多个字符从两端移动,向中间汇聚 例如:
helloworld!************h**********he********d!hel******ld!.....
helloworld!
要实现上述过程,首先我们得创建两个字符串,一个字符串arr1里面放入我们的“hello world”,另外一个字符串arr2里面放入“************”,创建好两个字符串之后,要实现它们的移动,就是要将arr1中的字符一个一个移动到arr2中,所以得使用一个左下标和一个右下标,然后让arr1[左下标]替换掉arr2[左下标],替换完之后,左下标要加1,然后让arr1[右下标]替换掉arr2[右下标],然后右下标减1,每次要替换完之后都将arr打印出来,直到左下标大于右下标
//while循环实现#include <string.h>#include <stdio.h>intmain()
{
//设置两个字符串chararr1[] ="hello world!";
chararr2[] ="************";
//左下标intleft=0;
//右下标//右下标是整个字符串的总长度减1intright=strlen(arr2) -1;
//当左下标大于右下标时循环停止while (left<=right)
    {
//实现交换arr2[left] =arr1[left];
arr2[right] =arr1[right];
//打印printf("%s\n", arr2);
//改变下标,继续打印left++;
right--;
    }
return0;
}

这段代码实现了多个字符从两端移动,向中间汇聚,但是在运行之后,在执行窗口里面是直接弹出结果,可以加点代码使它在运行之后,打印一行之后休息一秒钟让结果是一行一行弹出来,这样的效果就比较直接
#inlcude <windows.h>Sleep(1000);
这是Windows代码,在使用时要包含头文件<windows.h> 表示的是让代码休息某一段时间然后再开始执行 (时间)这里的时间表示毫秒,设置1000毫秒刚好是一秒

代码优化:

#include <string.h>#include <stdio.h>#include <windows.h>intmain()
{
//设置两个字符串chararr1[] ="hello world!";
chararr2[] ="************";
//左下标intleft=0;
//右下标//右下标是整个字符串的总长度减1intright=strlen(arr2) -1;
//当左下标大于右下标时循环停止while (left<=right)
    {
//实现交换arr2[left] =arr1[left];
arr2[right] =arr1[right];
//打印printf("%s\n", arr2);
//打印一行之后休息一秒钟Sleep(1000);
//改变下标,继续打印left++;
right--;
    }
return0;
}
各位老铁可以自己试一下效果 我们还可以再进行优化一下,让字符串的移动在一行上实现,不会换行打印,因此我们还可以再加点代码,来让它在一行慢慢的移动,因此当打印完一行之后,我们需要将屏幕清空,这时就需要用到我们在关机小程序中提到的命令提示符-清空屏幕
system("cls");  //清空屏幕

代码优化:

#include <string.h>#include <stdio.h>#include <windows.h>intmain()
{
//设置两个字符串chararr1[] ="hello world!";
chararr2[] ="************";
//左下标intleft=0;
//右下标//右下标是整个字符串的总长度减1intright=strlen(arr2) -1;
//当左下标大于右下标时循环停止while (left<=right)
    {
//实现交换arr2[left] =arr1[left];
arr2[right] =arr1[right];
//打印printf("%s\n", arr2);
//打印一行之后休息一秒钟Sleep(1000);
//打印完一行之后清空屏幕system("cls");
//改变下标,继续打印left++;
right--;
    }
//由于全部打印完之后会清理掉屏幕,所以可以在让它打印一下printf("%s\n", arr2);
return0;
}

这样子写使得代码在一行实现多个字符从两端移动,向中间汇聚,各位老铁也可以在自己电脑上试一下,很有趣 上面代码是使用while循环来写的,我们也可以使用for循环 将初始化,判断条件,调整部分都放进for循环中

代码演示:

//for循环实现#include <string.h>#include <stdio.h>#include <windows.h>intmain()
{
//设置两个字符串chararr1[] ="hello world!";
chararr2[] ="************";
//左下标intleft=0;
//右下标intright=0;
//当左下标大于右下标时循环停止for (left=0, right=strlen(arr2) -1; left<=right; left++, right--)
    {
//实现交换arr2[left] =arr1[left];
arr2[right] =arr1[right];
//打印printf("%s\n", arr2);
//打印一行之后休息一秒钟Sleep(1000);
//打印完一行之后清空屏幕system("cls");
    }
//由于全部打印完之后会清理掉屏幕,所以可以在让它打印一下printf("%s\n", arr2);
return0;
}

本期分享到此结束,如果文章有什么疑问或者不足,请大家分享在评论区或者私信,感谢大家学习!

目录
相关文章
|
5月前
|
C语言
C语言分支和循环语句
分支语句由`if-else`构成,用于根据不同条件执行相应代码。`else`会与最近未配对的`if`结合,多个条件可用`else if`实现。若连续使用`if`,各条件互不影响。嵌套结构可在`if`中再加入`if-else`。此外,`switch`语句适用于多分支选择(注意表达式不能为浮点数,`case`后需加`break`)。循环语句包括`for`、`while`和`do...while`,注意`do...while`末尾需加分号。循环中,`break`直接终止循环,`continue`跳过当前循环剩余部分,但`for`的语句三仍会执行,而`while`中位置影响效果。
75 0
|
9月前
|
人工智能 Java 程序员
一文彻底搞清楚C语言的循环语句
本文介绍了C语言中的三种循环语句:`while`、`do-while`和`for`,并详细解释了它们的语法格式、执行流程及应用场景。此外,还讲解了循环控制语句`break`和`continue`的使用方法。希望这些内容能帮助你在编程道路上不断进步,共同成长!
740 0
一文彻底搞清楚C语言的循环语句
|
10月前
|
C语言
【C语言程序设计——循环程序设计】枚举法换硬币(头歌实践教学平台习题)【合集】
本文档介绍了编程任务的详细内容,旨在运用枚举法求解硬币等额 - 循环控制语句(`for`、`while`)及跳转语句(`break`、`continue`)的使用。 - 循环嵌套语句的基本概念和应用,如双重`for`循环、`while`嵌套等。 3. **编程要求**:根据提示在指定区域内补充代码。 4. **测试说明**:平台将对编写的代码进行测试,并给出预期输出结果。 5. **通关代码**:提供完整的代码示例,帮助理解并完成任务。 6. **测试结果**:展示代码运行后的实际输出,验证正确性。 文档结构清晰,逐步引导读者掌握循环结构与嵌套的应用,最终实现硬币兑换的程序设计。
146 19
|
10月前
|
算法 C语言
【C语言程序设计——循环程序设计】求解最大公约数(头歌实践教学平台习题)【合集】
采用欧几里得算法(EuclideanAlgorithm)求解两个正整数的最大公约数。的最大公约数,然后检查最大公约数是否大于1。如果是,就返回1,表示。根据提示,在右侧编辑器Begin--End之间的区域内补充必要的代码。作为新的参数传递进去。这个递归过程会不断进行,直到。有除1以外的公约数;变为0,此时就找到了最大公约数。开始你的任务吧,祝你成功!是否为0,如果是,那么。就是最大公约数,直接返回。
247 18
|
10月前
|
Serverless C语言
【C语言程序设计——循环程序设计】利用循环求数值 x 的平方根(头歌实践教学平台习题)【合集】
根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码,求解出数值x的平方根;运用迭代公式,编写一个循环程序,求解出数值x的平方根。注意:不能直接用平方根公式/函数求解本题!开始你的任务吧,祝你成功!​ 相关知识 求平方根的迭代公式 绝对值函数fabs() 循环语句 一、求平方根的迭代公式 1.原理 在C语言中,求一个数的平方根可以使用牛顿迭代法。对于方程(为要求平方根的数),设是的第n次近似值,牛顿迭代公式为。 其基本思想是从一个初始近似值开始,通过不断迭代这个公式,使得越来越接近。
211 18
|
10月前
|
C语言
【C语言程序设计——循环程序设计】统计海军鸣放礼炮声数量(头歌实践教学平台习题)【合集】
有A、B、C三艘军舰同时开始鸣放礼炮各21响。已知A舰每隔5秒1次,B舰每隔6秒放1次,C舰每隔7秒放1次。编程计算观众总共听到几次礼炮声。根据提示,在右侧编辑器Begin--End之间的区域内补充必要的代码。开始你的任务吧,祝你成功!
202 13
|
10月前
|
存储 C语言
【C语言程序设计——循环程序设计】利用数列的累加和求 sinx(头歌实践教学平台习题)【合集】
项的累加和,一般会使用循环结构,在每次循环中计算出当前项的值(可能基于通项公式或者递推关系),然后累加到一个用于存储累加和的变量中。在C语言中推导数列中的某一项,通常需要依据数列给定的通项公式或者前后项之间的递推关系来实现。例如,对于一个简单的等差数列,其通项公式为。的级数,其每一项之间存在特定的递推关系(后项的分子是其前项的分子乘上。,计算sinx的值,直到最后一项的绝对值小于。为项数),就可以通过代码来计算出指定项的值。对于更复杂的数列,像题目中涉及的用于近似计算。开始你的任务吧,祝你成功!
244 6
|
10月前
|
C语言
【C语言程序设计——循环程序设计】鸡兔同笼问题(头歌实践教学平台习题)【合集】
本教程介绍了循环控制和跳转语句的使用,包括 `for`、`while` 和 `do-while` 循环,以及 `break` 和 `continue` 语句。通过示例代码详细讲解了这些语句的应用场景,并展示了如何使用循环嵌套解决复杂问题,如计算最大公因数和模拟游戏关卡选择。最后,通过鸡兔同笼问题演示了穷举法编程的实际应用。文中还提供了编程要求、测试说明及通关代码,帮助读者掌握相关知识并完成任务。 任务描述:根据给定条件,编写程序计算鸡和兔的数量。鸡有1个头2只脚,兔子有1个头4只脚。
519 5
|
C语言
【c语言】分支语句
C语言通过三种基本结构——顺序、选择和循环,构建复杂的程序逻辑。本文主要介绍了C语言的选择结构,即if-else语句及其变体,包括简单的if语句、if-else组合、else if多分支判断、嵌套if以及解决悬空else问题的方法。此外,还详细讲解了逻辑运算符和关系运算符的使用,以及如何利用条件操作符简化逻辑判断。最后,文章对比了if-else与switch语句在实现多分支逻辑时的应用,并解释了switch语句中的break和default关键字的作用。
187 8
|
C语言
【c语言】循环语句
循环结构是C语言中用于简化重复操作的重要工具,主要包括while循环、do-while循环和for循环。while循环是最基本的形式,通过不断检查条件来决定是否继续执行循环体。do-while循环则先执行循环体,再检查条件,至少执行一次。for循环逻辑更复杂,但使用频率最高,适合初始化、条件判断和更新变量的集中管理。此外,循环中还可以使用break和continue语句来控制循环的提前终止或跳过当前迭代。最后,循环可以嵌套使用,解决更复杂的问题,如查找特定范围内的素数。
202 6