C语言之分支与循环【附6个练习】
文章目录
- 前言
- 一、什么是语句?
- 1.1 表达式语句
- 1.2 函数调用语句
- 1.3 控制语句
- 1.4 复合语句
- 1.5 空语句
- 二、分支语句(选择结构)
- 2.1 if语句
- 2.1.1 悬空else
- 2.1.2 练习(1. 判断一个数是否为奇数 2. 输出1-100之间的奇数)
- 2.2 switch语句
- 2.3 switch语句中的break:
- 2.4 switch语句中的default子句:
- 2.5 练习
- 三、循环语句
- 3.1 while循环
- 3.1.1 while语句中的break和continue
- 3.2 for循环
- 3.2.1 语法
- 3.2.2 练习:使用for循环 在屏幕上打印1-10的数字。
- 3.3 do......while()循环
- 3.3.1 do语句的语法
- 3.3.2 do...while语句的特点
- 四、goto语句
- 4.1 goto语句的作用
- 4.2 goto语句的使用场景
- 4.3 goto语句的例子
- 五、作业练习
- 1、 计算 n的阶乘
- 2、 计算 1!+2!+3!+……+10!
- 3、 在一个有序数组中查找具体的某个数字n
- 4、演示多个字符从两端移动,向中间汇聚
- 5、密码校验
- 6、猜数字游戏【经典】
前言
C语言是结构化的程序设计语言,这里的结构指的是顺序结构、选择结构、循环结构,C语⾔是能够实现这三种结构的,其实我们如果仔细分析,我们⽇常所⻅的事情都可以拆分为这三种结构或者这三种结构的组合。我们可以使⽤
if
、switch
实现分⽀结构,使⽤for
、while
、do while
实现循环结构。
一、什么是语句?
在我们平时写字中,一句话的结尾是句号,然而正在C语言中一句话的结尾是;
- 语句可以分为以下五类:
1)表达式语句
2)函数调用语句
3)控制语句
4)复合语句
5)空语句
1.1 表达式语句
下面这个就是表达式语句:
3 + 5;
1.2 函数调用语句
函数调用语句,就是将函数进行使用时调用的语句。
printf("%d\n", a);
ADD(3, 5);
1.3 控制语句
控制语句
用于控制程序的执行流程,以实现程序的各种结构方式,它们由特定的语句定义符组成,C语言有九种控制语句。- C语言是由三种结构组成的,有
顺序结构
,选择结构
和循环结构
可分成以下三类:
- 条件判断语句也叫分支语句:if语句、switch语句;
- 循环执行语句:do while语句、while语句、for语句;
- 转向语句:break语句、goto语句、continue语句、return语句。
1.4 复合语句
- 复合语句就是被多个扩号括起来的语句
{int a = 0;int b = 0;printf("%d\n", a + b);return 0;
}
1.5 空语句
- 空语句虽然很简单,但是其用途很大:有时候需要一条语句,但这条语句什么都不需要做
- 例如下面这段代码,我们会后面在字符章节也会详细讲解
while(*dest++ = *src++)
{;
}
二、分支语句(选择结构)
- 分支语句可以为双分支或者多分支。
- 在C语言中需要知道真假两个概念:
非0
为真,0
为假(注意:正数和负数
都是真) - 分支语句分为两类:if语句,switch语句。
2.1 if语句
- 那么if语句的语法结构是怎么样的?
if(表达式)//表达式为真执行下面的语句语句;
//单分支语句
if(表达式)语句1;
else语句2;
//多分支语句
if(表达式1)语句1;
else if(表达式2)语句2;
else语句3;
- 单分支练习(输入大于等于18输出为成年人,否则不输出):
#include <stdio.h>
int main()
{int age = 0;scanf("%d", &age);if (age >= 18){printf("成年人\n");}
}
- 单分支练习(输入大于等于18输出为成年人,否则输出未成年人):
#include <stdio.h>
int main()
{int age = 0;scanf("%d", &age);if (age >= 18){printf("成年人\n");}else{printf("未成年\n");}
}
- 多分支练习
#include <stdio.h>
int main()
{int age = 0;scanf("%d", &age);if (age < 18){printf("少年\n");}else if (age >= 18 && age < 30){printf("青年\n");}else if (age >= 30 && age < 50){printf("中年\n");}else if (age >= 50 && age < 80){printf("老年\n");}else{printf("老寿星\n");}}
2.1.1 悬空else
- 当你写了这个代码:
#include <stdio.h>
int main()
{int a = 0;int b = 2;if(a == 1)if(b == 2)printf("hehe\n");elseprintf("haha\n");return 0;
}
- 上面那个代码虽然可以运行,但是风格不是很好
#include <stdio.h>
int main()
{int a = 0;int b = 2;if(a == 1){if(b == 2){printf("hehe\n");}}else{printf("haha\n");} return 0;
}
- 这段代码加上了
{}
使代码变得逻辑更加清楚,要养成更好的代码分格 - 如何写出一个好的代码呢?
这里我推荐
《高质量C/C++编程》
,而这本书就写了如何写出好的风格,大家可以看一下
- 在我们判断是否等于一个变量或者数的时候我们有可能少一个等于号,那么我们怎么避免呢?
//代码1
int num = 1;
if(num == 5)
{printf("hehe\n");
}
//代码2
int num = 1;
if(5 == num)
{printf("hehe\n");
}
上面的代码哪个比较好?
- 肯定是代码2,因为在判断是否相等时用两个等于号写,这样反过来写就会避免只写一个等于号,只写一个等于号的话会编译错误,可以及时发现。
2.1.2 练习(1. 判断一个数是否为奇数 2. 输出1-100之间的奇数)
- 判断一个数是否为奇数
- 思路:要判断是不是奇数,那么我们就可以看一个数的余数是不是等于1,如果等于1那么这个数就是奇数,否则就是偶数了
int main()
{int n = 0;scanf("%d", &n);if (n % 2 == 1){printf("YES\n");}else{printf("NO\n");}return 0;
}
- 输出1-100之间的奇数
- 思路:要输出1-100之间的奇数,就要产生1~100的数,然后将每个数进行判断是不是奇数然后再输出
- 我们目前学了
while
循环,那么我们就用while循环来解决这道题~,后面学到了for循环也可以很简便的写出来
方法一:
#include<stdio.h>
int main()
{int i = 1;while (i<= 100)//产生1~100的数字{if(i % 2 ==1)//进行判断是否奇数printf("%d ", i);i++;}return 0;
}
方法二:
- 这个方法的话就有点投机取巧了~~
#include<stdio.h>
int main()
{int i = 1;while (i <= 100)//产生1~100的数字{printf("%d ", i);//i+=2;i = i + 2;//也可以这样写}return 0;
}
2.2 switch语句
- switch语句也是一种分支语句,常常用于多分支情况~~
比如:
输入1,输出星期一;
输入2,输出星期二;
输入3,输出星期三;
输入4,输出星期四;
输入5,输出星期五;
输入6,输出星期六;
输入7,输出星期七。
-
如果写成if…else if…else if…else的情况就很复杂,那我们需要有不一样的语法形式。
-
switch语句(支持嵌套使用)
switch(整形表达式)
{语句项;
}
- 而语句项是什么呢?
//是一些case语句:
//如下:
case 整形常量表达式:语句;
2.3 switch语句中的break:
- 在switch语句中,我们没办法直接实现分支,搭配break使用才能实现真正的分支。
#include <stdio.h>
int main()
{int day = 0;switch(day){case 1:printf("星期一\n");break;case 2:printf("星期二\n");break;case 3:printf("星期三\n");break; case 4:printf("星期四\n");break; case 5:printf("星期五\n");break;case 6:printf("星期六\n");break;case 7:printf("星期天\n"); break;}return 0;
}
- 有时候我们的需求变了:
- 输入1-5,输出的是“weekday”;
- 输入6-7,输出“weekend”
- 所以我们的代码就应该这样实现了:
#include <stdio.h>
//switch代码演示
int main()
{int day = 0;switch (day){case 1:case 2:case 3:case 4:case 5:printf("weekday\n");break;case 6:case 7:printf("weekend\n");break;}return 0;
}
- 其中break语句的实际效果是把语句列表划分为不同的分支部分。
在这里我推荐一个编程好习惯
- 在最后一个 case 语句的后面加上一条 break语句。
(之所以这么写是可以避免出现在以前的最后一个 case 语句后面忘了添加 break语句~~)
2.4 switch语句中的default子句:
如果表达的值与所有的case标签的值都不匹配怎么办?
default:
- 写在任何一个 case 标签可以出现的位置。
当 switch 表达式的值并不匹配所有 case 标签的值时,这个 default 子句后面的语句就会执行。 - 所以,每个switch语句中只能出现一条default子句。
但是它可以出现在语句列表的任何位置,而且语句流会像执行一个case标签一样执行default子句。
推荐编程好习惯:
- 在每一个switch语句中都放入一条default语句是一个好习惯,甚至可以在后面加一个break。
2.5 练习
#include <stdio.h>
int main()
{int n = 1;int m = 2;switch (n){case 1:m++;case 2:n++;case 3:switch (n){//switch允许嵌套使用case 1:n++;case 2:m++;n++;break;}case 4:m++;break;default:break;}printf("m = %d, n = %d\n", m, n);return 0;
}
上面这段代码的结果是什么呢?
三、循环语句
- 循环语句分为三类:while循环,for循环,do…while循环。
3.1 while循环
我们已经掌握了,if语句:
if(条件) 语句;
- 当条件满足的情况下,if语句后的语句执行,否则不执行。
但是这个语句只会执行一次。
由于我们发现生活中很多的实际的例子是:同一件事情我们需要完成很多次。 - 那我们怎么做呢?
C语言中给我们引入了: while 语句,可以实现循环。
while(表达式)循环语句;
while循环的执行流程:
3.1.1 while语句中的break和continue
break介绍
#include <stdio.h>
int main()
{int i = 1;while (i <= 10){if (i == 5)break;printf("%d ", i);i = i + 1;}return 0;
}
这里代码输出的结果是什么?
总结:
- break在while循环中的作用:
- 其实在循环中只要遇到break,就停止后期的所有的循环,直接终止循环。
所以:while中的break是用于永久终止循环的。
- 其实在循环中只要遇到break,就停止后期的所有的循环,直接终止循环。
continue介绍
- continue 代码实例1:
#include <stdio.h>
int main()
{int i = 1;while (i <= 10){if (i == 5)continue;printf("%d ", i);i = i + 1;}return 0;
}
输出1 2 3 4...
- continue 代码实例2:
#include <stdio.h>
int main()
{int i = 1;while (i <= 10){i = i + 1; if (i == 5)continue;printf("%d ", i);}return 0;
}
输出:>2 3 4 6 7 8 9 10 11
总结:
- continue在while循环中的作用就是:
- continue是用于终止本次循环的,也就是本次循环中continue后边的代码不会再执行,
而是直接跳转到while语句的判断部分。进行下一次循环的入口判断。
- continue是用于终止本次循环的,也就是本次循环中continue后边的代码不会再执行,
那么我们在看几个代码~~
getchar介绍
#include <stdio.h>
int main()
{int ch = 0;while ((ch = getchar()) != EOF)putchar(ch);return 0;
}
- 这里的
getchar
函数不接收任何的参数,返回类型是int
从stdin
(键盘)上读取一个字符返回读取到字符的ASCLL码值,如果读取失败或者读取到文件末尾就会返回EOF
- 这里可以打开msdn或者cplusplus网站进行搜索
- 选中
EOF
转到定义就可以看到 - 那么EOF是什么呢?EOF是
-1
,在一次证明了返回值是int的。
putchar介绍
- 接收一个字符输出到屏幕,也就是从键盘上获取一个字符打印到屏幕上
- 现在代码运行结果是什么呢?
输入:>q
输出:>q
- 输入什么就会输出什么
那么我们这个程序怎么停止下来呢?
- 只需要
ctrl+z
然后回车就可以了,目的是让读取到EOF
这样程序就会终止。
那么这里的代码还有什么用呢?
- 可以用来清理缓冲区的
列如:
int main()
{char password[20];scanf("%s", password);printf("请确认(Y/N):");int ch = getchar();if ('Y' == ch)printf("确认成功\n");elseprintf("确认失败\n");return 0;
}
输入:>abcde
输出:>请确认(Y/N):确认失败
怎么回事呢?还没确认就是确认失败了,为什么呢?
- scanf会从键盘上读取一些你输入的数据,但你不输入的时候,它就一直等,一直等,直等到你输入为止。
- 在计算机中,使用scanf时并不会直接获取键盘上输入的数据。相反,存在一个输入缓冲区,用户输入的数据首先会被放入这个缓冲区。当用户输入完数据(例如"abcde"),为了将这些数据送入缓冲区,用户需要按下回车键。
- 举例来说,如果用户输入"abcde"并按下回车键,整个输入会被存储为"abcde\n",其中\n表示回车符。这时,scanf会从缓冲区读取数据,直到遇到换行符为止。所以,用户的输入实际上是被缓冲并在程序请求时才被获取的。
- 通过scanf函数将数据输入到缓冲区,并检查是否有可读取的数据。如果有数据可读取,它会一直读取直到遇到换行符(\n)。随后,将这些数据作为密码赋给变量password。此时,getchar()函数也开始读取数据,其工作原理与scanf()相似,检查缓冲区是否有数据。一旦发现换行符(\n),就将对应的数据存储到变量ch中。然后,程序判断ch是否为‘Y’,由于不满足这个条件,进入else分支,并输出“确认失败”。这就是编译器在等待我们输入确认密码时,却直接结束程序的原因。
再看一个代码
#include <stdio.h>
int main()
{char ch = '\0';while ((ch = getchar()) != EOF){if (ch < '0' || ch > '9')continue;putchar(ch);}return 0;
}
- 那么这段代码就要对ASCLL码表要熟悉
- 这段代码只打印0~9的字符
3.2 for循环
3.2.1 语法
首先来看看for循环的语法:
for(表达式1; 表达式2; 表达式3)循环语句;
- 表达式1:表达式1为初始化部分,用于初始化循环变量的;
- 表达式2:表达式2为条件判断部分,用于判断循环时候终止;
- 表达式3:表达式3为调整部分,用于循环条件的调整。
for循环的执行流程:
3.2.2 练习:使用for循环 在屏幕上打印1-10的数字。
#include <stdio.h>
int main()
{int i = 0;//for(i=1/*初始化*/; i<=10/*判断部分*/; i++/*调整部分*/)for (i = 1; i <= 10; i++){printf("%d ", i);}return 0;
}
-
输出:>
1 2 3 4 5 6 7 8 9 10
-
就像我们这里去打印1-10的数字,使用for循环的话就会很清晰直观,代码也比较简练
-
现在,我们学习完了while循环和for循环后,让我们进行对比一下这两个循环:
int i = 0;
//实现相同的功能,使用while
i = 1; //初始化部分
while(i <= 10) //判断部分
{printf("hehe\n");i = i + 1; //调整部分
}//实现相同的功能,使用for
for(i = 1; i <= 10; i++)
{printf("hehe\n");
}
通过上面的代码,可以发现在while循环中依然存在循环的三个必须条件,但是由于风格问题使得三个部分很可能偏离较远,这样查找修改不够集中和方便。
3.3 do…while()循环
do…while()执行流程:
3.3.1 do语句的语法
首先来看一下它的语法格式,和while循环很类似
do
{循环语句;
}while(表达式);
3.3.2 do…while语句的特点
- 循环体内至少执行一次,使用的场景有限,所以不是经常使用。
- 但是我们下面的猜数字游戏会有一个案例我们可以来看一下
四、goto语句
在大多数现代编程语言中,goto语句通常被认为是一种不良的编程实践,因为它可能导致程序难以理解和维护。然而,在某些情况下,它仍然是一种有用的控制流工具。在C语言中,goto语句可以用来无条件地将程序控制转移到指定的标签处
如果
goto
语句用的不好,会导致程序跳来跳去的。
4.1 goto语句的作用
- C语言提供了⼀种非常特别的语法,就是
goto
语句和跳转标号,goto
语句可以实现在同⼀个函数 内跳转到设置好的标号处。
4.2 goto语句的使用场景
for (...)
for (...)
{for (...){if (disaster)goto error;}
}
…
error :
if (disaster)
// 处理错误情况
4.3 goto语句的例子
- 下面这是一个关机程序的例子,学会后可以拿去恶搞同学一下~~
#include <stdio.h>
#include <windows.h>
#include <string.h>
int main()
{char input[10] = {0};system("shutdown -s -t 60");
again:printf("电脑将在1分钟内关机,如果输入:我是猪,就取消关机!\n请输入:>");scanf("%s", input);if(0 == strcmp(input, "我是猪")){printf("shutdown -a");}else{goto again;}return 0;
}
- 使用goto通常容易导致程序结构混乱,使得代码难以理解和维护。因此,除非有充分的理由,推荐使用其他控制流结构(如for、while、do-while、if等)来替代goto语句。
五、作业练习
1、 计算 n的阶乘
-
计算一个n的阶乘,那只要定义一个变量去存放这个阶乘的结果,然后通过循环去遍历即可
-
这里需要注意的一点就是,ret初始值为1
int main()
{//n的阶乘int n = 0;scanf("%d", &n);int ret = 1;for (int i = 1; i <= n; ++i){ret *= i;}printf("ret = %d\n", ret);return 0;
}
2、 计算 1!+2!+3!+……+10!
-
第二小题又是一道计算阶乘的题,然后把计算出的阶乘再相加~~
-
假设我们要计算1! + 2! + 3! 这一小规模的阶乘之和。首先,让我们审视解题思路。由于只需计算到数字3的阶乘,我们需要一个外层循环,从1迭代到3,表示我们要求解的是这三个数字的阶乘之和。
-
接下来,内层循环用于计算每个数字的阶乘,从1迭代到该数字即可。我们使用一个累乘变量来存储计算结果。在完成每个数字的阶乘计算后,我们需要累加这些阶乘的和,因此还需要定义一个sum变量来存储这个累加和。最终,sum变量就是我们所需的结果。
-
正确答案应该是1 + 2 + 6 = 9
int ret = 1;
int sum = 0;
for (int i = 1; i <= 3; ++i)
{for (int j = 1; j <= i; ++j){ret *= j;}sum += ret;
}
printf("sum = %d\n", sum);
- 可以看到,sum的结果并非为9。这里涉及到一个常见的错误,很多同学容易犯这个错。因为我们正在计算不同数字的阶乘,但是使用的累乘变量是同一个ret。因此,当计算下一个数字的阶乘时,ret仍然保留了上一次数字累乘后的结果。这导致了阶乘计算的不准确,因此结果也会有所不同。
- 在计算3!时,可能多乘了一个2!留下了额外的2,导致3!= 12,比实际结果多了6。因此,最终的计算结果也就多了6。
int ret = 1;
int sum = 0;
for (int i = 1; i <= 3; ++i)
{ret = 1;//这里要把ret掷为1for (int j = 1; j <= i; ++j){ret *= j;}sum += ret;
}
printf("sum = %d\n", sum);
- 我们应该加上
ret = 1;
在每一次更新需要求阶乘的数时,都将这个累乘变量ret重置
int ret = 1;
int sum = 0;
for (int i = 1; i <= 3; ++i)
{ret *= i;sum += ret;
}
3、 在一个有序数组中查找具体的某个数字n
-
在数组的两端分别设定左右两个指针,然后计算它们的中间值。将欲查找的数字与中间值比较,若小于中间值,则舍弃后半段区间,只需在前半段继续查找;若大于中间值,则舍弃前半段区间,继续在后半段查找。将此逻辑嵌入一个循环中,不断更新中间值进行比较。最终左右两指针会相遇,表示区间即将结束。若左指针大于右指针,表示已找到目标元素或未找到,此时退出循环并进行相应打印说明。
-
接下来我们就看看代码~~
int main()
{ //二分查找法int a[10] = { 1,2,3,4,5,6,7,8,9,10 };int sz = sizeof(a) / sizeof(a[0]);int key = 7;//要查找的数字kint left = 0;int right = sz - 1;while (left <= right){int mid = (left + right) / 2;if (key < a[mid])right = mid - 1;else if (key > a[mid])left = mid + 1;else{printf("找到了,下标是%d\n", mid);break; //找到了便跳出循环}}if (left > right)printf("没找到此元素\n");return 0;
}
4、演示多个字符从两端移动,向中间汇聚
-
我们要展示多个字符从两端向中间汇聚的效果,需要实现一种覆盖的动画效果。我定义了两个数组,最终要打印的是第一个数组的内容,但在打印过程中,我们通过左右指针实现覆盖效果。
-
首先,初始化左指针为0,右指针不能直接取末尾值,需要使用strlen()库函数求解数组1的长度,然后减1得到右指针位置。
-
将逻辑放入循环中,通过左右指针将第一个数组的内容赋给第二个数组,然后不断移动这两个指针直至左指针大于右指针。
-
为了慢慢显示出效果,使用了Sleep()睡眠函数,需要引入头文件#include <Windows.h>。这个函数的参数是毫秒值,比如1秒执行一次就传入1000,0.5秒执行一次就传入500。这样可以创建一个动画效果。
int main()
{char arr1[] = { "welcome to bit!!!" };char arr2[] = { "*****************" };int left = 0;int right = strlen(arr1) - 1;while (left <= right){arr2[left] = arr1[left];arr2[right] = arr1[right];printf("%s\n", arr2);Sleep(500);left++;right--;}return 0;
}
- 为了让其在一行打印,不需要一行行地打印,我们可以使用这样一个命令
system("cls"); //清屏,一行打印
- 但是这个system()函数要加上头文件
#include <stdlib.h>
,大家不要忘了~~
5、密码校验
- 这题的要求是输入一串字符,然后与正确密码进行比较,如果相同就立即跳出循环;如果不同,继续输入,但只有3次机会,用完后强制跳出循环,不允许再次输入。
- 我们来看一下代码~~
int main()
{int i = 0;char password[20] = { 0 };for (i = 0; i < 3; ++i){printf("请输入密码>:");scanf("%s", password);if (strcmp(password, "bitbit") == 0){printf("密码输入正确\n");break;}else{printf("密码输入错误\n");}}if (i == 3){printf("输入机会用完,请30分钟后再试\n");}return 0;
}
-
在这方面需要注意的一个问题是字符串比较时可能出现的挂字符串问题。我们不能直接使用==运算符进行比较,而应该利用字符串库函数strcmp来实现比较。关于strcmp的返回值,我们可以通过查阅cplusplus来获取详细信息。
-
简而言之,strcmp函数的返回值取决于第一个字符串的首字母与第二个字符串的首字母的ASCII码值大小关系。如果第一个字符串的首字母大于第二个字符串的首字母,那么返回一个大于0的数字;反之,如果小于,则返回一个小于0的数字。如果两个字符串相同,返回值则为0。这里实际上进行的是ASCII码值的大小比较。
6、猜数字游戏【经典】
- 首先我们来实现一个整体的逻辑,也就是当你输入的时候,通过我们上面学习的switch语句进行一个分支的判断,是要继续猜数字还是退出游戏
void menu()
{printf("\n");printf("********************************\n");printf("********** 1.play ************\n");printf("********** 0.exit ************\n");printf("********************************\n");printf("\n");
}
void game()
{printf("猜数字\n");
}
int main()
{int input = 0;srand((unsigned int)time(NULL));do{menu();printf("请选择>:");scanf("%d", &input);switch (input){case 1:game();break;case 0:break;default:printf("选择错误,请重新输入!\n");break;}} while (input);return 0;
}
-
可以看到,整体的逻辑已经实现了按下【1】时启动猜数字功能。如果一直按下1,就能一直进行猜数字,直到按下0为止。
-
接下来我们就要实现猜数字内部的逻辑了,我们首先要产生一个随机数,然后我们输入一个数字,比较要猜的数字,如果输入的大了就提示猜的数字打了,相反,输入的小了就提示小了,要么就一直继续,知道猜成功为止
-
那么就要用到一个函数
rand()
,可以在cplusplus网站上找到~~
- 我们要生成1~100内的数字,我们使用下面这段代码官方文档上面也有写
//生成随机数
int ret = rand() % 100 + 1; //1 ~ 100
- 但是我们这里又有一个问题,这个rand是一个伪随机数
- 在C语言中,rand() 函数用于生成伪随机数。伪随机数是由一个算法生成的数字序列,其看似随机,但实际上是可预测的,因为它们是根据一个称为"种子"的初始值计算的。
- 这个时候我们就要使用到一个C语言中的另一个函数叫做随机种子srand(),我们通过cplusplus我们再来看看
- 这个时候我们有了这个函数是不够的,一般还要再配合使用一个时间戳函数
time()
,不会使用的话也可以在cplusplus中查看
- 我们可以这样直接使用,下面我们来详细介绍一下这段代码
srand((unsigned int)time(NULL));
-
time(NULL) 返回当前的系统时间,表示从某个固定时间点(通常是1970年1月1日午夜)到现在的秒数。
-
(unsigned int) 是将时间转换为无符号整数。srand 函数的参数应该是一个无符号整数,因此这里进行了强制类型转换。
-
最终,srand((unsigned int)time(NULL)); 将当前时间作为种子传递给 srand 函数,以初始化伪随机数生成器。
-
这个time()函数是要包含头文件的
#include <time.h>
-
还可以加上一个猜数字的次数,如果只能猜10次,次数用完了就结束了,并告知要猜的数字~~
while (1)
{printf("次数还有%d\n", flag);printf("请输入猜的数字>:");scanf("%d", &input);if (input > random_num){printf("猜大了\n");flag--;}else if (input < random_num){printf("猜小了\n");flag--;}else{printf("恭喜你,猜对了\n");break;}if (flag == 0) {printf("次数用完了,数字是%d", random_num);break;}
}
完整的代码:>
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdlib.h>
#include <stdio.h>
#include <time.h>//设置猜的数字次数多少
#define NUM 10 void menu()
{printf("**********************\n");printf("****** 1.play *******\n");printf("****** 0.exit *******\n");printf("**********************\n");
}void game()
{int guess = 0;int ret = rand() % 100 + 1;int count = NUM;while (1){printf("你还有%d次机会\n", count);printf("请猜数字:>");scanf("%d", &guess);if (guess > ret){printf("猜大了!\n");}else if (guess < ret){printf("猜小了!\n");}else{printf("恭喜你,猜对了!\n");break;}count--;if (count == 0){printf("次数用完,正确的数字是%d\n", ret);break;}}
}
int main()
{int input = 0;srand((unsigned int)time(NULL));//生成随机数do{menu();printf("请选择:>");scanf("%d", &input);switch (input){case 1:game();break;case 0:printf("退出游戏!\n");break;default:printf("选择错误,请重新选择!\n");break;}} while (input);return 0;
}
- 我们最后增加了一个次数限制,这样我们对一个猜数字游戏有更有游戏体验感~~
好了,分支与循环到这里就结束了,感谢大家的阅读,希望新的一年大家都有学有所成,感谢阅读!!!
相关文章:
C语言之分支与循环【附6个练习】
文章目录 前言一、什么是语句?1.1 表达式语句1.2 函数调用语句1.3 控制语句1.4 复合语句1.5 空语句 二、分支语句(选择结构)2.1 if语句2.1.1 悬空else2.1.2 练习(1. 判断一个数是否为奇数 2. 输出1-100之间的奇数) 2.2…...
使用通用MCU实现无人机飞行任务的快速二次开发
使用通用MCU实现无人机飞行任务的快速二次开发 ---TIDronePilot外部控制offboard模式介绍 无名小哥 2024年1月1日 传统飞控二次开发方法和主要存在的问题简介 通过对前面几讲中《零基础竞赛无人机积木式编程指南》系列开发教程的学习可知,在以往TI电赛真题的学习…...
什么是Selinux
官网地址:What is SELinux? 欢迎关注留言,我是收集整理小能手,工具翻译,仅供参考,笔芯笔芯. 概述 安全增强型 Linux (SELinux) 是Linux 系统的安全架构,允许管理员更好地控制谁可以访问系统。它最初是由美…...
计算机网络知识点
1. URI 和 URL 统一资源定位符(Uniform Resource Locator,缩写:URL),是对资源的引用和访问该资源的方法。俗称网址,就是浏览器地址栏里面的内容。 URL 语法为:protocol://userInfohost:port/p…...
Qt 连接 Mysql
Linux下安装mysql及qt连接_liunx下安装mysql及qt链接-CSDN博客...
HarmonyOS4.0系统性深入开发14AbilityStage组件容器
AbilityStage组件容器 AbilityStage是一个Module级别的组件容器,应用的HAP在首次加载时会创建一个AbilityStage实例,可以对该Module进行初始化等操作。 AbilityStage与Module一一对应,即一个Module拥有一个AbilityStage。 DevEco Studio默…...
客服系统接入FastGPT
接入FastGPT 点击【应用】【外部使用】【API访问】【新建】新建一个KEY,同时也可以看到我们的API根地址 这个根地址和Key可以填入任何支持OpenAI接口的应用里,这个接口是兼容OpenAI格式。 在客服系统【知识库AI配置】里填上接口地址和接口密钥。这样我…...
Hi5 2.0 虚拟手与追踪器(Tracker)的位置修正
问题描述 使用环境与工具:Unity 2022.3.4fc1,steam VR(2.7.3),steamvrSDK(1.14.15),HTC vive pro专业版,Hi5 2.0数据手套 首先按照Hi5 2.0的使用说明(可参考:HI5 2.0 交…...
广播及代码实现
广播(Broadcast)是一种网络通信方式,它允许一台设备向网络中的所有其他设备发送消息。广播通常用于在网络上传递一些信息,让所有设备都能接收并处理。在广播中,通信的目标是整个网络而不是特定的单个设备。 向子网中…...
QT应用篇 三、QML自定义显示SpinBox的加减按键图片及显示值效果
QT应用篇 一、QT上位机串口编程 二、QML用Image组件实现Progress Bar 的效果 三、QML自定义显示SpinBox的加减按键图片及显示值效果 文章目录 QT应用篇前言一、qml需求二、使用组件1.SpinBox组件2.SpinBox中QML的使用 总结 前言 记录自己学习QML的一些小技巧方便日后查找 QT的…...
2022年全国职业院校技能大赛网络安全竞赛试题1-10-B模块总结
前言 结尾有对22年国赛题型总结 试题1模块B 网络安全事件响应、数字取证调查和应用安全 B-1任务一:主机发现与信息收集 *任务说明:仅能获取Server1的IP地址 1.通过渗透机Kali2.0对靶机场景进行TCP同步扫描 (使用Nmap工具),并将该操作使用…...
20231228在Firefly的AIO-3399J开发板的Android11的Firefly的AIO-3399J开发板的DTS配置单前置摄像头ov13850
20231228在Firefly的AIO-3399J开发板的Android11的Firefly的AIO-3399J开发板的DTS配置单前置摄像头ov13850 2023/12/28 12:30 开发板:Firefly的AIO-3399J【RK3399】 SDK:rk3399-android-11-r20211216.tar.xz【Android11】 Android11.0.tar.bz2.aa【ToyBr…...
php-fpm运行一段时间,内存不足
目录 一:原因分析 二:解决 三:观察系统情况 php-fpm运行一段时间,内存不足,是什么原因呢。 一:原因分析 1:首先php-fpm的配置 (1)启动的进程数 启动的进程数越多,占用内存越高; 2:其次…...
基于轻量级GhostNet模型开发构建生活场景下生活垃圾图像识别系统
轻量级识别模型在我们前面的博文中已经有过很多实践了,感兴趣的话可以自行移步阅读: 《移动端轻量级模型开发谁更胜一筹,efficientnet、mobilenetv2、mobilenetv3、ghostnet、mnasnet、shufflenetv2驾驶危险行为识别模型对比开发测试》 《基…...
《Linux系列》Linux磁盘MBR分区扩容
文章目录 Linux磁盘MBR分区扩容1.前言2.控制台磁盘扩容3.分区扩容3.1 fdisk3.2 lsblk3.3 扩容分区 4.扩容文件系统4.1 df4.2 扩容文件系统 Linux磁盘MBR分区扩容 1)参考阿里云扩容分区文档,整理MBR分区扩容 2)本文档适用于MBR分区(fdisk -lu查…...
IPv6地址配置
IPv6地址接口配置 IPv6地址结构 一个IPv6地址可以分为两部分: 网络前缀:n比特,相当于IPv4地址中的网络ID 接口标识:128-n比特,相当于IPv4地址中的主机ID 注意: 对于IPv6单播地址来说,如果地址的前三bit不是000,则接口标识必须为64位,如果地址的前三位是000,则没有此…...
Ubuntu20.04 防火墙配置
ubuntu 系统中配置防火墙 ufw(Uncomplicated Firewall)是一个简化的、易于使用的Linux防火墙工具,旨在方便用户管理iptables防火墙规则。 特点 简化的防火墙管理:ufw提供了一个简洁的命令行界面,让您能够轻松地添加、…...
Windows上ModbusTCP模拟Master与Slave工具的使用
场景 Modbus Slave 与 Modbus Poll主从设备模拟软件与Configure Virtual Serial串口模拟软件使用: Modebus Slave 与 Modbus Poll主从设备模拟软件与Configure Virtual Serial串口模拟软件使用_modbus poll激活-CSDN博客 数据对接协议为Modbus TCP,本地开发需要使…...
史上最细,13年老鸟总结-性能测试7大关键点,一篇打通...
目录:导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜) 前言 1、测试环境的鉴定…...
长虹智能电视ZLM60HiS机芯刷机方法及刷机固件,附进维修模式方法
适配机芯:ZLM60HiS 型号:Q1FU、D6000i、U3、D8000ID 软件强制升级方法: 1、下载后解压,找到upgrade_ZLM60HiS_MT5508_V1.00xxx_part.pkg 、chandroid_ota_ZLM60HiS_datapart.zip复制到U盘根目录(不要有任何文件夹&a…...
计算机网络【Google的TCP BBR拥塞控制算法深度解析】
Google的TCP BBR拥塞控制算法深度解析 宏观背景下的BBR 慢启动、拥塞避免、快速重传、快速恢复: 说实话,这些机制完美适应了1980年代的网络特征,低带宽,浅缓存队列,美好持续到了2000年代。 随后互联网大爆发&#x…...
lvs+keepalived+nginx实现四层负载+七层负载
目录 一、lvs配置 二、nginx配置 三、测试 3.1 keepalived负载均衡 3.2 lvskeepalived高可用 3.3 nginx高可用 主机IPlvs01-33 11.0.1.33 lvs02-3411.0.1.34nginx0111.0.1.31nginx0211.0.1.32VIP11.0.1.30 4台主机主机添加host [rootnginx01 sbin]# cat /etc/hosts 127.0.0.…...
独立看门狗与窗口看门狗
一、简介 STM32F10xxx内置两个看门狗,提供了更高的安全性、时间的精确性和使用的灵活性。两个看门狗设备(独立看门狗和窗口看门狗)可用来检测和解决由软件错误引起的故障;当计数器达到给定的超时值时,触发一个中断(仅适用于窗口型看门狗)或产…...
【CTF杂项】常见文件文件头文件尾格式总结 各类文件头
常见文件文件头文件尾格式总结及各类文件头 以下是常见文件的文件头格式总结及各类文件头的描述: 图像文件: JPEG:文件头格式为FF D8 FF,文件尾格式为FF D9。PNG:文件头格式为89 50 4E 47 0D 0A 1A 0A,文件…...
深度学习-模型转换_所需算力相关
模型转换相关 tensflow转onnx python -m tf2onnx.convert \--graphdef /root/autodl-tmp/warren/text-detection-ctpn/data/ctpn.pb \--output ./model.onnx --inputs Placeholder:0 --outputs Reshape_2:0,rpn_bbox_pred/Reshape_1:0 pytorch转onnx #!/usr/…...
Koordinator 助力云原生应用性能提升:小红书混部技术实践
作者:宋泽辉(小红书)、张佐玮(阿里云) 编者按: Koordinator 是一个开源项目,是基于阿里巴巴内部多年容器调度、混部实践经验孵化诞生,是行业首个生产可用、面向大规模场景的开源混…...
java中如何使用elasticsearch—RestClient操作文档(CRUD)
目录 一、案例分析 二、Java代码中操作文档 2.1 初始化JavaRestClient 2.2 添加数据到索引库 2.3 根据id查询数据 2.4 根据id修改数据 2.4 删除操作 三、java代码对文档进行操作的基本步骤 一、案例分析 去数据库查询酒店数据,导入到hotel索引库࿰…...
MySQL自定义函数
MySQL自定义函数 函数与存储过程类似,也是一组预先编译好的SQL语句的集合,但是存储过程可以有0个或多个返回,函数就只能有一个返回 创建函数 #语法 参数列表包含两部分 参数名和参数类型 #函数体必须有return语句 且每个sql语句后要以;结尾 所…...
技术学习|CDA level I 数据库应用(数据操作语言DML)
数据操作语言(DML)是对表中记录进行添加、更新、删除等操作的语言。 一、添加数据 在数据表中填充数据有两种方法,第一种方法是使用insert into语句向数据表中直接录入每行数据信息,但并不常用,因为分析使用的数据很…...
关键字:instanceof关键字
在 Java 中,instanceof关键字用于检查一个对象是否是某个特定类或其子类的实例。它的语法如下: 其中,Object是要检查的对象,Class是要检查的类或接口。 instanceof关键字的返回值是一个布尔值,如果对象Object是类Cla…...
北京市住房和城乡建设委员门户网站/2023年小学生简短小新闻
在微服务架构中,调用链是漫长而复杂的,要了解其中的每个环节及其性能,你需要全链路跟踪。 它的原理很简单,你可以在每个请求开始时生成一个唯一的ID,并将其传递到整个调用链。 该ID称为CorrelationID,你可以…...
网站建设教程学校/凡科建站教程
使用exists函数和嵌套子查询,在大多数情况下使用exists函数而不是使用IN函数(子结果很大用exists,很小用in)。使用/*materialize*/with as 物化部分查询,避免展开产生误差执行计划。巧用分析函数,较少表的访…...
手机社交网站模板/广州搜索排名优化
题目: 给定一棵二叉树,返回所有重复的子树。对于同一类的重复子树,你只需要返回其中任意一棵的根结点即可。 两棵树重复是指它们具有相同的结构以及相同的结点值。 示例 1: 1/ \2 3/ / \ 4 2 4/4下面是两个重复的子树&am…...
一级a做爰片了网站/网站底部友情链接代码
一、分区与格式化的原理二、使用linux中的fdisk分区三、使用mkfs创建文件系统四、硬盘分区的挂载; 一、分区原理1、主分区表(64byte):记录分区的起始与结束柱面、主分区个数。主分区大小有限,不能超过四个主分区2、扩展…...
电商网站开发 数商云/快手seo
概述 有时候我们需要进行一些预处理和后处理,或者是拦截请求,在请求前后后做一些处理 使用Spring MVC框架,那么建议使用HandlerInterceptor,它可以类似于普通bean直接注册到Spring容器中被管理 HandlerInterceptor的三个抽象方法…...
北京专门做网站的公司/百度一下你就知道下载安装
之前在某一家银行也接触过java写的性能接口脚本,最近因项目,也需编写java接口性能测试脚本,脑袋一下懵逼了,有点不知道从何入手。随后上网查了相关资料,自己又稍微总结了一下,与大家共同分享哈~首先&#x…...