C语言中分支与循环语句的示例分析
这篇文章主要介绍C语言中分支与循环语句的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!
- if语句:if(表达式)
// 括号里面放一个表达式
//表达式的结果如果为非零,表达式为真
//表达式结果如果为零,表达式为假
if语句可以单支,双支,多分支,还可以用大括号括起来之后执行多条语句,下图为双分支示例:
if(表达式) 语句;//每一个分号隔开的叫做一条语句else 语句;//注意!无大括号直接写只能执行一条语句
下图为多分支示例:
if(表达式)//第一条语句也没有分号 语句;else if(表达式); 语句;//注意else if语句结束要有分号else if(表达式); 语句: //此处省略若干个else ifelse//此处便是最后一条语句了,无分号 语句;
下图为执行多条语句的示例:
if(表达式){ 语句01; 语句02; //....}//下面的可以单支,双支也可以多分支
悬空else问题
示例:
#includeint main(){ int a = 0; int b = 2; if(a == 1) if (b == 2) printf("hehe\n"); else printf("haha\n"); return 0;}
以上这段代码的打印结果大多数人可能第一眼看到之后,就会说屏幕上会打印 :haha,因为他们会认为:第六行的表达式判断之后,a是不等于1的,故表达式为假,执行else后面的语句。但是其实不然,else只于离他最近的那个 if 匹配,因此以上的代码的真实打印结果为:不打印。
所以好的代码书写风格可以大大减少不必要的误会
//正确的书写方式if(a == 1) if(b == 2) printf("hehe\n"); else printf("haha\n");
- switch语句
switch语句也是一种分支语句,多用于多分支的情况
switch语句的语法格式:
switch(整形表达式)//后面无需再放分号{case (整形常量表达式): 语句; break;//break是决定了程序走到该位置之后还要不要往下走//有break直接跳出switch//无break继续往下执行其他的case语句//直至遇到break为止}
要注意的细节:
(以代码为例)
#includeint main(){ int day = 0; scanf("%d",&day); switch (day) { case 1: //以下略 } return 0;}
要注意的细节
(对照上图)
如果将第4行的代码改成:float day,那么改程序将无法继续执行,因为这样改完之后原来的day就被改为了浮点型,day传到第6行之后还是一个浮点型,而使用switch的语法明确规定:switch(),括号中要写整型常量表达式,必须为整型和常量
还有,如果在switch语句执行开头提前定义好一个整形变量,并给它赋值,之后再把这个变量放入case 的后面,此程序也是无法执行的。下图示例:
int n = 1;switch (/*此处略*/){ case n://上面的操作一定会引起编译器的报错//因为n本质上还是属于是一个变量//case后根据语法规范必须为整形且常量}
另外,如果case后面跟的满足整型,常量,并且是一个表达式的话,也是可以执行的下去的'
//示例case 1+0://这样写也是可以编译的
最后就是,如果case后面跟的是一个字符型也是能够编译过去的,因为字符也是属于整形的一种,字符以ascii码的形式储存在计算机之中的。
switch中的的default子句:
此子句适用于处理那些所有分支情况之外的输入
示例:
//多余的代码略写int day = 0;scanf("%d".&day);switch(day){ case 1://case内的语句和break略写 case 2: case 3: case 4: case 5: case 6: case 7: default: printf("输入错误\n"); break; }
如上图所示,case语句之中只给出了七个分支来选择,但是如果输入者一不小心输入错误,没有输入1至7中的数(比如输入了一个9),那么程序最后就会不编译,为防止出现这种情况的发生,所以专门设计了一个default子句用来供那些别的错误情况进入,以给予输入者一个错误提示。另外,default子句不管放在开头还是结尾都没问题,但是我们一般默认放置句尾。
循环语句:循环结构分为三种:while循环,for循环,do while循环
- while循环语法结构
while(//表达式,即判断循环执行的条件){ 循环语句;}
上面的表达式结果如果为真,即非0,那么循环执行
如果表达式结果为假,即为0 ,那么循环体不执行
注意事项:
如果在while循环中有break,那么该break用于调出当前所在的循环体
就是说只能跳一级,跳出它当前所在的循环。如果外面还有循环体,那照样还要继续执行下去
如果在while循环中有continue,那么continue的作用就是用于跳过continue后面的代码,直接到程序开头的判断部分,看要不要继续往下执行代码,示例如下:
//代码多余的部分略while (i <= 10){ if(i == 5) continue; printf("%d ",i); i++;}
上面的代码打印结果就是 1234,然后后面就不打了
因为当i变为5的时候经 if 语句的判断为真,到continue处,又回到代码while循环判断的开头,看是否执行下一次的循环,判断之后可执行,又经if 来到continue处,又回到while处判断,可执行…如此往复下去,没有跳出这个死循环的可能。后面也不可能再去打印别的东西。
for循环的continue,是直接跳到for循环的表达式3之中,执行调整部分,由此可见while循环和for循环的continue是有一定区别的,while循环是完全有可能直接跳过调整部分的,因为调整部分有可能在他的下面的代码
譬如上面的代码块就是这样的。
- for循环语法结构
for(表达式1;表达式2;表达式3){ 循环语句;}
for循环其实是while循环的进一步改进,因为while循环的初始化部分(int i = 0),条件判断部分(while(i>10)),调整部分(i ++),这几个部分之间相隔有的时候会很远,如果需要改动的话就会很容易改动。
for循环中,表达式1用于给循环变量一个初始值(表达式 1,只会执行一次,往后就没用了,每次循环开始前变量保留着上一次的值,除非这个循环本身就在另一个循环之中,这样的话每次都会初始化一遍,所以一般情况下,再上来的时候循环变量只会走 表达式 2 判断一下和 表达式 3 自增一下,随后继续去执行下面的代码),表达式2用于判断,表达式3用于调整。
要注意的要点:尽量不要在循环体内改变循环变量,不然循环很容易失去控制,以下图代码块为例:
#includeint main(){ for(i=0; i<11; i++) { printf("%d",i); i = 5; } return 0;}
上述代码是一个连续打印6的一个死循环,因为每次走到第七行时,i 都会被赋值成 5。再上去到调整语句自增。
所以说循环内部千万不要改变循环变量。
其次要注意的要点:for循环的语句判断部分采用开区间,因为使用开区间的话不等号旁边表示的是循环的次数
代码的可读性更高。比如:i<11.
for循环的用法也是非常灵活的,三个表达式可以随意省略,但是判断表达式还是最好不要省略,因为极易造成死循环。下面再来看一个示例:
#includeint main(){ int i = 0; int j = 0; for(; i<3;i++) for(; j<3; j++) { printf("hehe\n"); } return 0;}
此处按照正常的理解来说,第四五行应该已经对 i 和 j 进行了初始化,随后再去执行的时候,循环出三个 i ,每个 i 再分出来三个 j,所以总共应该打印应该是九个hehe。但事实上并不是这样,真正的执行的过程为:i 初始化为 0之后,i 加一,与 j 相关的循环执行三次,printf 也打印三次,但是之后 i 在加一次,在执行 j 循环时,j 的初始化是在循环之外的,因此 j 仍然是3,循环无法执行,在往后也一样是这样。所以最终只打印三次hehe。
for循环还可以使用两个变量循环变量来控制,例子如下:
int x ,y;for(x = 0&&y = 0;x<10&&y<10;x++&&y++)
- do while循环
do{ 语句1; 语句2; .......}while(判断语句);//判断语句旁边的的分号千万不能少,不然的话语法就过不去了
do while循环执行方式很简单,就是什么都不管直接先去执行括号内的语句,执行完了再去判断还要不要继续走下去,所以说do while 最少都能执行一次
do while 语句中的 continue 与 break 的用法
continue:直接跳过它下面的语句至最后的那个判断语句,然后再从头开始
break:直接就是调出所在的那个循环不执行了
示例:
do{ if(i == 5) continue; printf("%d ",i); i++;}while(i<=10);
最终打印结果就是: 1234,因为当代码中 i 走到 5 的时候,第5 行的continue 就会开始执行,使其跳转到第8 行进行判断,判断为真,跳到开头的第4行,继续执行,再继续跳转…所以从 i 变成 5 开始,这段代码就变成了死循环。
此外,如果第 5 行的语句被改成了 break 就会直接跳出所在的循环,直接去执行第 9 行的语句
循环练习题
- 题目一(阶乘)
//计算n的阶乘#includeint main(){ int ret = 1; int i = 0; int n = 0; scanf("%d",&n); for(i = 1; i<=n; i++) { ret *= i; } printf("%d",ret); return 0;}
- 题目二(阶乘和)
//计算 1!+2!+3!+.....+10!//只写出了核心步骤 int ret = 1; int i,sum = 0; int n = 1; for(n=1; n<11; n++) { ret = 1;//ret每次都必须要重新赋值//不然里面会保留上一次的阶乘值//致使我们无法形成阶乘相加的效果 for(i = 1; i<=n; i++)//注意,这里的 i ,每次都会初始化一次//因为这个i的循环本身,就是在循环里面 { ret *= i; } sum += ret; } return 0;
上面这段代码写出来,思路非常的加单粗暴,就是要有各个不同的阶乘,那么就用 n 来控制,要有阶乘,那么就用 i 来控制。但是这种思路用来解决阶乘相加的问题其实效率过低,因为每次当代码走到第14行的时候 ,每次要算阶乘的时候总是会出现一个问题,就是每一次阶乘前面总是会把上一次的阶乘重算一遍,再乘上一个最新更新的 i ,这样非常浪费时间
图解上面这段代码的缺陷:
所以说我们可以这样想,可不可以更高效地利用数据,比如:1! 用完不要扔掉,再给他乘以一个 2,
变成 2!,同理 2!也不要扔掉,再给他乘以一个 3,最后把他们相加放到 sum 里面就行了。
//计算 1!+2!+3!+......+10!//优化后的核心步骤for(n=1; n<11; n++){ ret *= n; sum += ret;}
优化后的图解:
可以对照其对应的代码块,不难发现,这个效果一个循环就搞定了,利用率还非常高
- 题目三(二分查找)
//在一个有序数组中查找具体的某个数//有序,即排好序的,如果从前往后找的话效率过低//此题使用二分查找法#includeint main(){ int arr[10] = {1,2,3,4,5,6,7,8,9,10}; int k = 7;//此处为要查找的元素 int sz = size of(arr)/size of(arr[0]); //上面这行是用来计算出数组里面的元素个数 int left = 0; int right = sz - 1; while(left <= right) { int mid = (left + right)/2; if(arr[mid] < k) { left = mid + 1; } else if(arr[mid] > k) { right = mid - 1; } else { printf("找到了,下标为:%d\n",mid); break; } } if(left > right) { printf("找不到\n"); } return 0;}
使用二分查找法去寻找数可以大大提高查找效率,其原理非常简单:就是先算出数组中的元素个数,然后将查找时移动的右下标(right)初始值表示出来(即:数组元素个数减一),其次左下标(left)的初始值赋 0 (因为数组元素下标是从 0 开始的),然后表示出 mid ,就是左右下标的平均值,再用这个平均值去和要去查找的数 k 去比较,如果 mid 比 k 大,mid减一并将值赋给left,实现左值的更新,若 mid 比 k 小,mid 减一赋给 right,实现右值的更新,由此不断循环往复左右值相距越来越近,若能实现左右值相等,则查找的数是存在的,若逼近到最后直至左右值交叉,则说明要查找的数不存在。图示如下:
[
https://imgtu.com/i/gm5KGF
- 题目四(两边往中间渐变)
//编写多个字符从两端向中间移动//其实就是一串字符从两端向中间移动//每次露出左数第一个未知和右数第一个未知#include#include #include int main(){ char arr1[] = "welcome to world"; char arr2[] = "################"; int left = 0; int right = strlen (arr1)-1; printf("%s\n", arr2); while(left <= right) { arr2[left] = arr1[left]; arr2[right] = arr1[right]; Sleep(1000);//此处s必须大写,这里起到休眠1秒的作用 system("cls");//此处起到闪屏的动态作用 left++; right++; printf("%s",arr2); } return 0;}
这道题目的原理和二分查找法类似,不再赘述。
此外还要注意的是:strlen()用于求字符串的长度,即里面含有几个字符
sizeof()用于求数组所占的内存空间的大小
- 题目五(密码登录)
//模拟密码登录的情景,只允许登录三次//三次都不对,则退出程序#include#include int main(){ int i = 0; char password[20] = { 0 }; for (i = 0; i < 3; i++) { printf("请输入密码:> "); scanf("%s", password);//数组名本身就是地址,不是需要再用取地址符 if (strcmp(password, "123456") == 0)//if后别加分号!!(淦,找了一个下午的bug) { printf("登陆成功\n"); break; } else { printf("登陆失败,请重新输入!\n"); } } if (i == 3) { printf("三次错误,退出程序\n"); } return 0;}
此处的 strcmp(字符串 1,字符串 2)是用于对比字符串的内容是否一致,如果一致,strcmp 返回一个 0 ,这也是为什么 if 里面判断其是否等于 0。如果不一致,有两种情况:字符串 1 大于字符串 2 ,返回值大于 0 ;字符串 1 小于字符串 2,返回值小于 0 。此外,strcmp 的比较方法是:字符串上的每一位逐个比较,比较对应位置上字符的 ASCII 码值,不是比较字符串的长度,一旦对应位置上的字符大小分出高下,那么两个字符也立刻分出高下,不管后面的字符或大或小或长或短。下面用图示来解释:
- 题目六(猜数字游戏)
//写一个猜数字游戏//游戏要求如下://1.自动产生一个 1-100 之间的随机数//2.猜数字// a.猜对了,就恭喜你游戏结束// b.你猜错了,会告诉你猜大了,还是猜小了//3.游戏会一直玩,除非退出游戏#include#include #include void menu()//menu函数的功能就是去实现一个菜单界面{ printf("-------------------\n"); printf("-----1.玩游戏------\n"); printf("-----2,退出游戏----\n"); printf("-------------------\n");}void game()//game函数的功能就是去实现游戏的基本操作和随机数的产生{ int ret = 0; int guess = 0; ret = rand() % 100 + 1;//此处就是生成随机数的关键,rand的用法可以去MSDN去查 //还有题目要求是1-100的随机数所以rand返回值必须模上100.随机数就可以被限定在0-99再加1,就是1-100 while (1)//这个while实现的就是猜数字的的过程 { printf("请猜一个数:> \n"); scanf("%d", &guess); if (guess < ret)//这里就是要让guess这个猜的值和我们随机生成的数来做比较 { printf("猜小了\n"); } else if(guess > ret ) { printf("猜大了\n"); } else { printf("恭喜你,猜对了\n"); break; } }}int main(){ srand((unsigned int)time(NULL));//这里和上面的rand相联系,是rand语法规定的一部分,这里的作用来给生成的随机数一个起点 //里面的那个time()是个时间戳,就是用现在的时间和计算机的起始时间换算出一个随机数 //因为srand里面必须要填上一个随机数所以用时间就是再好不过的了 //还有就是srand的语法规定里面的数必须是无符号整形所以(unsigned int)用来强制改变数的类型 int input = 0; do //此处使用do while循环就是要保证不管玩不玩先进去选择 { menu(); printf("请按菜单要求输入一个数:> "); scanf("%d", &input); switch (input) { case 1: game(); break; case 0: printf("退出游戏\n"); break; defult ://此处的的defult 的作用也很关键。它可以防止玩家输入其他数字进入到while()的判断之中被误判为真 printf("输入错误,请重输\n"); } } while (input);//此处用input来判断是很巧妙的,因为选择上input的只有0和1去供你选择,若为0便是假 跳出循环 system("pause"); //若为 1便是真,继续循环 //选择 0 的时候程序便会跳到67行终止 return 0;}
- goto语句
goto语句在C语言可以改变程序的进程顺序,只要在想跳转的地方做个标记,然后程序走到对应的goto语句处就会跳转到标记处重新开始执行。(感觉有点像《火影忍者》里面波风水门的飞雷神)但是写程序的时候要小心,goto语句有可能导致程序混乱产生bug。另外,goto语句不可以跨函数跳跃。
#includeint main(){flag://这里就是标记的地方,一开始程序进去忽略这个东西 printf("hehe\n"); printf("haha\n"); goto flag;//这里就是让程序跳转到标记处的语句 return 0;}
- getchar与putchar用法
下图为示例:
#includeint main(){ int ch = getchar(); //getchar是用来接收键盘上输入的字符 //之后将接受的字符转化为ASCII码值存放起来 //由前面的存储类型是int也是可以看出来的 //总之,用法是和scanf是差不多的 putchar(ch); //putchar把接收到的字符打印出来 //原理和printf差不多 return 0;}
此外还要注意的是:getchar获取一个字符如果失败了,那么他就会返回一个EOF
EOF 翻译过来就是:文件终止(end of file)
#includeint main(){ int ch = 0; while((ch = getchar())!=EOF) //上面条代码就是让getchar读取一个数 //判断它有没有读取错误,若对的话则循环 { putchr(ch); } return 0;}
上面的这段代码的效果即输入什么就打印什么,可以不停地输,不停地打
例:输 a 则打 a,输 b 则打 b。
如果想要循环停止的话,那么就可以输入一个:ctrl 加 z ,其原理其实就是给 getchar 输入一个错误字符
让他终止,返回一个 EOF。
getchar 接收字符的原理:getchar 和 电脑键盘之间有一个缓冲区,该缓冲区用于存放键盘上输入的东西,此外getchar 只要是缓冲区里面的东西一律照收不误,包括回车,回车如果输入进去就会被缓冲区理解为一个 \n ,所以上述代码的输入和打印形式为:
a //这里先给他一个a ,再打了一个回车
a
b
b
就是说,getchar 先读取了 a ,随后 a被打印,之后又读取一个 \n 由此 b 只能在第三行开始输入,以此类推…
getchar 与putchar的实际应用:
实例如下图:
#includeint main(){ int password[20] = {}; printf("请输入密码:> "); scanf("%s", password); printf("请确认密码:> "); int tmp = 0; while ((tmp = getchar()) != '\n') { ; } int ch = getchar(); if (ch == 'Y') { printf("确认成功\n"); } else { printf("确认失败\n"); } return 0;}
上面的一段的代码写的是一个输入密码并确认的函数,具体分析:
首先我们创建一个数组,用于存放输入的字符串,之后创建一个 tmp ,用循环来消耗 scanf 拿不走的字符串,就是让while里面的getchar在读到 \n 之前,一直用上面代码块中的第11行的空语句来来消耗那些没用的(就是说getchar读入,但不会进行操作),即空格剩下的 (空格)abcdef \n,因为如果不去消耗的话,第13行的getchar将直接读取 6 后面的那个空格,然后自动默认走else 后面的分支,随后在屏幕上直接打印 :确认失败。由此可见,getchar 的作用非常直白,只是在读取的时候看到缓冲区有什么就读什么。
[
以上是"C语言中分支与循环语句的示例分析"这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注行业资讯频道!