C++研发笔记16——C语言程序设计初阶学习笔记14

        本篇笔记我们继续来学习第三部分《分支语句和循环语句》,在上一篇中我们详细学习了循环语句——for语句以及循环语句——do-while循环,接下来本部分的内容还包括:折半查找算法、猜数字游戏、goto语句。本篇笔记便可以结束第三部分的知识学习部分。

第三部分  分支语句和循环语句

七、折半查找算法

在上一个目题中有这样一道习题:在一个有序数组中查找具体的某个数字n。除了之前给出的答案,还有另外一种更加简单的写法。

#include<stdio.h>
int main(){int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };int a = 0;printf("请输入一个整数:");scanf("%d", &a);int low = 0;int high = 9;int flag = 0;int mid = 0;while (low <= high){mid = (low + high) / 2;if (arr[mid] > a)high = mid - 1;else if (arr[mid] < a)low = mid + 1;else{printf("找到了 %d, 下标为 %d\n", a, mid);flag = 1;break;}}if (0 == flag)printf("找不到\n");return 0;
}

        这段代码使用到了折半查找的算法(二分算法),这是一种在有序数组中查找特定元素的高效搜索算法。它的基本思想是将目标值与数组中间元素进行比较,根据比较结果缩小搜索范围,然后继续在新的范围内进行查找,直到找到目标值或搜索范围为空。

以下是折半查找算法的基本步骤:

(1)初始化:设置两个变量,low 是数组的最低索引(通常是 0),high 是数组的最高索引(通常是数组长度减 1)。

(2)循环条件:当 low 小于等于 high 时,执行循环体。

(3)计算中间索引:计算中间索引 mid,通常为 (low + high) / 2。

(4)比较中间元素:将数组中间位置的元素与目标值进行比较。

(5)如果中间元素等于目标值,查找成功,返回中间索引 mid。

(6)如果中间元素大于目标值,说明目标值位于数组的左半部分,更新 high 为 mid - 1。如果中间元素小于目标值,说明目标值位于数组的右半部分,更新 low 为 mid + 1。

(7)重复:重复步骤 3 和 4,直到找到目标值或 low 大于 high。

(8)查找失败:如果 low 大于 high,说明数组中不存在目标值,返回一个表示失败的标志。

        学会了这个算法,便可以解决一些实际问题了,比如我现在希望猜出一双300元以下的鞋子的价格,实现代码如下:

#include <stdio.h>
void bin_search(int arr[], int size , int low , int high , int target) {high = size - 1;int mid = 0;while (low <= high) {mid = low + (high - low) / 2;if (arr[mid] == target) {printf("价格为%d", arr[mid]);break;}else if (arr[mid] < target) {low = mid + 1; // 目标值在右半部分}else {high = mid - 1; // 目标值在左半部分}}
}
int main() {int arr[300]; for (int i = 0; i < 300; i++) {arr[i] = i + 1; // 初始化数组}int target=250;bin_search(arr, 300, 0 ,299 ,target); // 调用二分查找函数return 0;
}

八、猜数字游戏

        掌握了前面学习的这些知识,我们就可以写一些比较实用有趣的代码了,现在我们就来编写一个“猜数字游戏”,游戏要求为:

· 电脑自动生成1~100的随机数。 

· 玩家猜数字,猜数字的过程中,根据猜测数据的大小给出大了或小了的反馈,直到猜对,游戏结束。

· 游戏可以反复进行。直到用户主动退出。

(一)、随机数生成

        我们想要实现这个猜数字游戏,首先就要产生一个随机数,那么实现生成随机数的方法是什么呢?

1、rand 函数

        C语言提供了一个函数叫 rand,这函数是可以生成随机数的,函数原型如下:

int rand(void);

        rand函数会返回一个伪随机数,这个随机数的范围是在0~RAND_MAX之间,这个RAND_MAX的大小是依赖编译器上实现的,但是大部分编译器上RAND_MAX的值都是32767。

        rand函数的使用需要包含一个头文件是:stdlib.h

        那我们就来测试一下rand函数,这里多调用几次,产生5个随机数:

#include <stdio.h>
#include <stdlib.h>
int main()
{printf("%d\n", rand());printf("%d\n", rand());printf("%d\n", rand());printf("%d\n", rand());printf("%d\n", rand());return 0;
}

        当我们多次运行这段代码可以看到虽然一次运行中产生的5个数字是相对随机的,但是下一次运行程序生成的结果和上一次一模一样,这就说明我们在使用这个函数时候是有问题的。

        如果再深入了解一下,我们就不难发现,其实rand函数生成的随机数是伪随机的,伪随机数不是真正的随机数,是通过某种算法生成的随机数。真正的随机数的是无法预测下一个值是多少的。而rand函数是对一个叫“种子”的基准值进行运算生成的随机数。

        之所以前面每次运行程序产生的随机数序列是一样的,那是因为rand函数生成随机数的默认种子是1。 如果我们想要要生成不同的随机数,就要让种子是变化的。

2、srand 函数

        C语言中又提供了一个函数叫 srand,用来初始化随机数的生成器的,srand的原型如下:

void srand(unsigned int seed);

        程序中在调用 rand 函数之前先调用 srand 函数,通过 srand 函数的参数seed来设置rand函数生成随机数的时候的种子,只要种子在变化,每次生成的随机数序列也就变化起来了。

        那也就是说给srand的种子如果是随机的,rand就能生成随机数;在生成随机数的时候又需要一个随机数,但是这就矛盾。

3、time 函数

        在程序中我们一般是使用程序运行的时间作为种子的,因为时间时刻在发生变化的。

        在C语言中有一个函数叫 time ,就可以获得这个时间,time函数原型如下:

time_t time(time_t* timer);

        time 函数会返回当前的日历时间,其实返回的是1970年1月1日0时0分0秒到现在程序运行时间之间的差值,单位是秒。返回的类型是time_t类型的,time_t 类型本质上其实就是32位或者64位的整型类型。

        time函数的参数 timer 如果是非NULL的指针的话,函数也会将这个返回的差值放在timer指向的内存中带回去。 如果 timer 是NULL,就只返回这个时间的差值。time函数返回的这个时间差也被叫做:时间戳。

        在使用 time函数的时候需要包含头文件:time.h

        当然目前我们还并没有学习指针的使用,所以大家不需要过分纠结于 time 函数的具体使用方式,在我们使用的时候,只需要返回时间戳就可以了,实现代码如下:

time(NULL);//调用time函数返回时间戳,这里没有接收返回值

        所以我们可以将生成随机数的代码修改如下:

#include <stdio.h>
#include <stdlib.h>
#include <time.h>
int main()
{
//使用time函数的返回值设置种子
//因为srand的参数是unsigned int类型,我们将time函数的返回值强制类型转换srand((unsigned int)time(NULL));printf("%d\n", rand());printf("%d\n", rand());printf("%d\n", rand());printf("%d\n", rand());printf("%d\n", rand());return 0;
}

        这段代码就可以实现随机生成不同的随机数了,另外我们要注意的一点是,srand函数是不需要频繁使用的,一次运行的程序中调用一次就足够了。

4、设置随机数的范围

        如果我们要生成0至99之间的随机数,方法如下:

rand() % 100;		//余数的范围是0至99

        如果我们要生成1至100之间的随机数,方法如下:

rand() % 100 + 1;		//%100的余数是0至99,0至99的数字+1,范围是1至100

        如果我们要生成100至200之间的随机数,方法如下:

100 + rand() % (200 - 100 + 1)		//余数的范围是0至100,加100后就是100至200

        根据上面的内容,我们可以总结出这样一个公式,及当我们想要生成a至b的随机数时,可以使用这样的公式:

a + rand() % (b - a + 1)

(二)、猜数字游戏实现

        了解了生成随机数字的逻辑之后,实现这段代码就简单多了,当然实现这个功能需要使用到函数的相关知识,同时这个功能对C语言初学者来讲还是较为困难的,所以大家可以暂时以理解实现这个功能的代码为目标,代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <time.h>
void menu()		//返回类型为void
{printf("**********************************************\n");printf("**********************************************\n");printf("**********************************************\n");printf("************  1  play(开始游戏) ************\n");printf("************  2  exit(退出游戏) ************\n");printf("**********************************************\n");printf("**********************************************\n");printf("**********************************************\n");
}
void game()		//返回类型为void —— 不需要返回
{srand((unsigned int)time(NULL));int r = rand() % 100 + 1;int guess = 0;while (1)		//这是一个死循环,除非使用break跳出{printf("请猜一个0至100之间的整数>:");scanf("%d", &guess);if (guess < r){printf("您猜小了,请重新输入\n");}else if (guess > r){printf("您猜大了,请重新输入\n");}else{printf("恭喜你,猜对了\n");printf("\n");break;}}
}
int main()
{int input = 0;do{menu();printf("\n请选择:>>");scanf("%d", &input);switch (input){case 1:game();break;case 2:printf("您选择了退出游戏,游戏结束\n");break;default:printf("输入错误,请从0和1之间选择\n");break;}}while (2 != input);return 0;
}

        当然这段代码值得改进的地方还有很多,比如说我可以加入一个猜数字次数的限制,如果6次猜不出来,就算挑战失败,添加功能后并优化运行结果视图效果的代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <time.h>
void menu()		//返回类型为void
{printf("**********************************************\n");printf("**********************************************\n");printf("**********************************************\n");printf("************  1  play(开始游戏) ************\n");printf("************  2  exit(退出游戏) ************\n");printf("**********************************************\n");printf("**********************************************\n");printf("**********************************************\n");
}
void game()		//返回类型为void —— 不需要返回
{srand((unsigned int)time(NULL));int r = rand() % 100 + 1;int guess = 0;int count = 6;while (count != 0){printf("你还有%d次机会\n", count);printf("请猜一个0至100之间的整数>:");scanf("%d", &guess);if (guess < r){printf("您猜小了,请重新输入\n");}else if (guess > r){printf("您猜大了,请重新输入\n");}else{printf("恭喜你,猜对了\n");printf("\n");break;}count -= 1;}if (count == 0){printf("很遗憾,六次机会使用完了,挑战失败!\n");printf("答案是%d", r);printf("\n");}
}
int main(){int input = 0;do{menu();printf("\n请选择:>>");scanf("%d", &input);printf("\n开始猜数字游戏啦!\n\n");switch (input){case 1:game();printf("\n请等待三秒钟将重新开始程序!\n");Sleep(3000);system("cls");break;case 2:printf("您选择了退出游戏,游戏结束\n");break;default:printf("输入错误,请从0和1之间选择\n");break;}} while (2 != input);return 0;
}

九、goto语句

        C语言中提供了可以随意使用的goto语句和标记跳转的标号。从理论上goto语句是没有必要的,实践中没有goto语句也可以很容易的写出代码。但是某些场合下goto语句还是用得到的,最常见的用法就是终止程序在某些深度嵌套的结构的处理过程,例如一次跳出两层或多层循环。这种情况使用break是达不到目的的。它只能从最内层循环退出到上一层的循环。

        下面是使用goto语句的一个例子:

#include<stdio.h>
int main()
{
flag:printf("hehe\n");goto flag;return 0;
}

        这个代码的运行结果会是一个不断打印”hehe”的死循环。过度使用 goto 语句会使代码难以理解和维护,因为它打破了程序的自然流程。

        当然goto语句也有使用的范围,goto语句只能在同一个函数内部跳转,不能实现跨越函数跳转。

        goto语句最实际的应用场景便是在跳出多重循环的场景,当然我们也可以使用循环、条件语句或其他控制流结构来替代 goto,从而使代码更加清晰。

for (...)
for (...){for (...){if (disaster)goto error;}
}
…
error :
if (disaster)

下面我们来看一个比较有意思的程序:

#include <stdio.h>
#include<string.h>
#include<stdlib.h>
int main(){char input[10] = { 0 };system("shutdown -s -t 60"); // 定时关机命令
again:printf("电脑将在1分钟内关机,如果输入:我是猪,就取消关机!\n请输入:>");scanf("%s", input);if (0 == strcmp(input, "我是猪")){system("shutdown -a"); //取消关机命令}else{goto again;}return 0;
}

        这段代码实现了在 Windows 系统中实现一个倒计时关机的交互式取消机制。但是我们不使用goto语句也可以实现同样的功能,因此我们在编程的过程中可以适当减少goto语句的使用。

#include <stdio.h>
#include<string.h>
#include <stdlib.h>
int main(){char input[10] = { 0 };system("shutdown -s -t 60");while (1){printf("电脑将在1分钟内关机,如果输入:我是猪,就取消关机!\n请输入:>");scanf("%s", input);if (0 == strcmp(input, "我是猪")){system("shutdown -a");break;}}return 0;
}

        以上便是本篇笔记的全部内容,我们继续了第三部分的学习,对折半查找算法、猜数字游戏以及goto语句进行了详细的讲解。到这里我们就结束了第三部分的知识学习的部分。在下一篇笔记中,我们会进行学习第三部分的练习。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/889278.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

视频推拉流EasyDSS无人机直播技术巡查焚烧、烟火情况

焚烧作为一种常见的废弃物处理方式&#xff0c;往往会对环境造成严重污染。因此&#xff0c;减少焚烧、推广绿色能源和循环经济成为重要措施。通过加强森林防灭火队伍能力建设与长效机制建立&#xff0c;各地努力减少因焚烧引发的森林火灾&#xff0c;保护生态环境。 巡察烟火…

K8S对接ceph的RBD块存储

1 PG数量限制问题 1.1 原因分析 1.还是老样子&#xff0c;先创建存储池&#xff0c;在初始化为rbd。 [rootceph141~]# ceph osd pool create wenzhiyong-k8s 128 128 Error ERANGE: pg_num 128 size 3 for this pool would result in 295 cumulative PGs per OSD (2067 tot…

React Router 6的学习

安装react-router-dom npm i react-router-dom 支持不同的路由创建 createBrowserRouter 特点 推荐使用的方式&#xff0c;基于 HTML5 的 History API。支持用户友好的 URL&#xff0c;无需 #。适用于生产环境的绝大多数场景。 适用 使用现代浏览器&#xff0c;支持 pus…

微信小程序web-view 嵌套h5界面 实现文件预览效果

实现方法&#xff1a;(这里我是在小程序里面单独加了一个页面用来下载预览文件) 安装 使用方法请参考文档 npm 安装 npm install weixin-js-sdk import wx from weixin-js-sdk预览 h5界面代码 <u-button click"onclick" type"primary" :loading"…

HTTP 状态码大全

常见状态码 200 OK # 客户端请求成功 400 Bad Request # 客户端请求有语法错误 不能被服务器所理解 401 Unauthorized # 请求未经授权 这个状态代码必须和WWW- Authenticate 报头域一起使用 403 Forbidden # 服务器收到请求但是拒绝提供服务 404 Not Found # 请求资源不存…

一文详解TCP协议 [图文并茂, 明了易懂]

欢迎来到啊妮莫的学习小屋! 目录 什么是TCP协议 TCP协议特点✨ TCP报文格式 三次握手和四次挥手✨ 可靠性 效率性 基于字节流✨ 基于TCP的应用层协议 什么是TCP协议 TCP(传输控制协议, Transmission Control Protocol) 是一种面向连接的, 可靠的, 基于字节流的传输层通…

在Linux(ubuntu22.04)搭建rust开发环境

1.安装rust 1.安装curl: sudo apt install curl 2.安装rust最新版 curl --proto ‘https’ --tlsv1.2 https://sh.rustup.rs -sSf | sh 安装完成后出现&#xff1a;Rust is installed now. Great! 重启当前shell即可 3.检验是否安装成功 rustc --version 结果出现&…

UnityShaderLab 实现程序化形状(一)

1.实现一个长宽可变的矩形&#xff1a; 代码&#xff1a; fixed4 frag (v2f i) : SV_Target{return saturate(length(saturate(abs(i.uv - 0.5)-0.13)))/0.03;} 2.实现一个半径可变的圆形&#xff1a; 代码&#xff1a; fixed4 frag (v2f i) : SV_Target{return (distance(a…

如何解决压测过程中JMeter堆内存溢出问题

如何解决压测过程中JMeter堆内存溢出问题 背景一、为什么会堆内存溢出&#xff1f;二、解决堆内存溢出措施三、堆内存参数应该怎么调整&#xff1f;四、堆内存大小配置建议 背景 Windows环境下使用JMeter压测运行一段时间后&#xff0c;JMeter日志窗口报错“java.lang.OutOfMe…

宽字节注入

尽管现在呼吁所有的程序都使用unicode编码&#xff0c;所有的网站都使用utf-8编码&#xff0c;来一个统一的国际规范。但仍然有很多&#xff0c;包括国内及国外&#xff08;特别是非英语国家&#xff09;的一些cms&#xff0c;仍然使用着自己国家的一套编码&#xff0c;比如gbk…

Web APIsPIs第1章

WebApi阶段学习什么&#xff1f; WebApi是浏览器提供的一组接口 使用 JavaScript 去操作页面文档 和 浏览器 什么是 API API: 应用程序接口&#xff08;Application Programming Interface&#xff09; 接口&#xff1a;本质上就是各种函数&#xff0c;无需关心内部如何实现…

android——录制屏幕

录制屏幕 1、界面 2、核心代码 import android.app.NotificationChannel import android.app.NotificationManager import android.app.PendingIntent import android.app.Service import android.content.Context import android.content.Intent import android.graphics.Bi…

【Excel学习记录】01-认识Excel

1.之前的优秀软件Lotus-1-2-3 默认公式以等号开头 兼容Lotus-1-2-3的公式写法&#xff0c;不用写等号 &#xff1a; 文件→选项→高级→勾选&#xff1a;“转换Lotus-1-2-3公式(U)” 备注&#xff1a;对于大范围手动输入公式可以使用该选项&#xff0c;否则请不要勾选&#x…

短视频矩阵抖音SEO源码OEM独立部署

短视频优化矩阵源码涉及对抖音平台上的视频内容进行筛选与排序&#xff0c;目的是增强其在搜索引擎中的可见度&#xff0c;以便更多用户能够浏览到这些视频。而抖音SEO优化系统则是通过构建一个分析框架&#xff0c;来解析抖音上的用户数据、视频信息及标签等元素&#xff0c;并…

MySQL——buffer poll

为什么要有buffer poll&#xff1f; 如果没有buffer poll&#xff0c;每次读取数据的时候都是从磁盘上读的&#xff0c;这样效率是很差的的。 所以有了提高效率的方式&#xff0c;就加上了一个缓存——buffer poll 所以&#xff0c;当我们读取数据的时候就有以下的方式 当读…

生产慎用之调试日志对空间矢量数据批量插入的性能影响-以MybatisPlus为例

目录 前言 一、一些缘由 1、性能分析 二、插入方式调整 1、批量插入的实现 2、MP的批量插入实现 3、日志的配置 三、默认处理方式 1、基础程序代码 2、执行情况 四、提升调试日志等级 1、在logback中进行设置 2、提升后的效果 五、总结 前言 在现代软件开发中&#xff0c;性能优…

元宇宙时代的社交平台:Facebook的愿景与实践

随着科技的不断进步&#xff0c;元宇宙&#xff08;Metaverse&#xff09;这一概念逐渐走进了人们的视野。作为全球最大的社交平台之一&#xff0c;Facebook&#xff08;现Meta&#xff09;在这场元宇宙革命中扮演着重要角色。Meta不仅在不断扩展其社交平台的边界&#xff0c;还…

C# 小案例(IT资产管理系统)

开发工具&#xff1a;visual studio 2022 语言&#xff1a;C# 数据库&#xff1a;Sql Server 2008 页面展示 一、登录 二、主窗体 三、用户管理 四、资产管理 五、关于 Java版地址&#xff1a;基于若依开发物品管理系统(springbootvue)_若依物品管理系统-CSDN博客 Python版…

分布式日志系统设计

一、分布式日志系统定义 分布式日志系统是一种用于收集、存储和分析大规模分布式系统日志的系统。它可以帮助开发人员和系统管理员实时监控和调试系统&#xff0c;提高系统可靠性和可用性&#xff0c;同时也可以用于日志分析和故障排查。 二、简单设计思路 日志收集&#xff…

敏捷开发04:Scrum 中的 Product Backlog(产品待办列表) 详细介绍

Product Backlog 产品待办列表 在计划开发产品功能时&#xff0c;都希望产品功能上线后&#xff0c;用户能够喜欢并经常使用。 因此在开发产品新功能时&#xff0c;就要衡量哪些产品需求是对用户最有价值&#xff0c;这是最应该思考的问题。 然后把这些有价值的需求集合放在一…