学C的第七天(继续 深入了解 循环语句,学习二分查找‘折半查找算法’;练习:计算n的阶乘、计算阶乘再相加、在有序数组中查找某个数字、多个字符从两端移动向中间汇聚、模拟用户登录情景)-2

简介: 练习:1.计算n的阶乘(不考虑溢出)

练习:

1.计算n的阶乘(不考虑溢出):

//计算 n的阶乘
#include <stdio.h>
//5!= 1 * 2 * 3 * 4 * 5 
//先用循环生成1-n个数字,再依次相乘
int main()
{
  int n = 0;
  scanf("%d", &n);
  int i = 0;
  int ret = 1; // 相乘后赋值给该函数,因为0乘任何数都为0,所以初始化为1
  for ( i = 1; i <= n; i++) // 利用循环生成1-n个数字
  {
    ret = ret * i; // 依次相乘
  }
  printf("%d\n", ret);
  return 0;
}


be0fb77f2e9c4a11a691d888790771e0.png

2.计算1! + 2! + 3! +...+ 10! :

第一种方法效率比较低,会重复计算之前的计算过的阶乘,但逻辑比较好理解

//计算1! + 2! + 3! +...+ 10! :
#include <stdio.h>
//5!= 1 * 2 * 3 * 4 * 5 
//内循环:循环生成1-n个数字,再依次相乘
//外循环:阶乘相加
int main()
{
  int n = 0;
  int i = 0;
  int ret = 1; // 相乘后赋值给该变量,因为0乘任何数都为0,所以初始化为1
  int sum = 0; // 外循环将阶乘相加赋给该值
  //外循环:阶乘相加
  for ( n = 1; i <= 10; n++) // 加到10!
  {
    ret = 1; // 内循环中相乘后赋值给该变量,每次出了循环要重新刷新该值(重点)
    //内循环:循环生成1-n个数字,再依次相乘
    for (i = 1; i <= n; i++) // 利用循环生成1-n个数字
    {
      ret = ret * i; // 依次相乘
    }
    sum += ret; // 将阶乘相加赋给该值
  }
  printf("%d\n", sum);
  return 0;
} 


image.png

第二种方法:在原本已计算的阶乘上再乘上’下一个数‘,实现‘下一个数’的阶乘)

//第二种方法:
//计算1! + 2! + 3! +...+ 10! :
#include <stdio.h>
//5!= 1 * 2 * 3 * 4 * 5 
int main()
{
  int n = 0;
  int i = 0;
  int ret = 1; // 相乘后赋值给该变量,因为0乘任何数都为0,所以初始化为1
  int sum = 0; // 将阶乘相加赋给该值
  for (n = 1; n <= 10; n++) // 执行到10!
  {
    ret = ret * n; // 直接乘n,
    //在原本已计算的阶乘上再乘上’下一个数‘,实现下一个数的阶乘
    sum += ret; // 将阶乘相加赋给该值
  }
  printf("%d\n", sum);
  return 0;
}


image.png

3.在一个有序数组中查找具体的某个数字n。(讲解二分查找算法‘折半查找算法’):

第一种方法逻辑简单,而且在无序数组中也可以使用,但效率较低,需要一个一个查找)

//在一个有序数组中查找具体的某个数字n。
#include <stdio.h>
int main()
{
  int arr[] = { 1,2,3,4,5,6,7,8,9,10 }; // 升序(有序数组)
  //数组下标:  0 1 2 3 4 5 6 7 8 9
  int find = 7; // 在数组中要找的值
  int i = 0; // 数组下标
  //利用for循环在数组中一个一个查找
  for ( i = 0; i < 10; i++ )
  {
    if (arr[i] == find) 
    {
      printf("找到了,该值在数组中对应的下标是:%d\n", i);
      break; // 找到后跳出循环
    }
  }
  if (i == 10) // 10超过数组下标了,说明数组中没有这个值
  {
    printf("数组中没有该值。\n");
  }
  return 0;
}

image.png

第二种方法二分查找算法‘折半查找算法’,在有序数组查找中效率更高,一次查找就能排除一半的值,重点!!)

思路:(重点!!))

改错:“对比中间下标和所找值的下标”,是要对比数组中对应下标的值


4c2c474712254cf7bd3c38a5386588a9.png

代码实现:结合上面原理图理解)

//第二种方法
#include <stdio.h>
int main()
{
  int arr[] = { 1,2,3,4,5,6,7,8,9,10 }; // 升序(有序数组)
  //数组下标:  0 1 2 3 4 5 6 7 8 9
  int find = 7; // 在数组中要找的值
  int i = 0; // 数组下标
  int sz = sizeof(arr) / sizeof(arr[0]); // 求元素个数
  // sizeof(arr): 计算数组总大小(单位是字节)
  // sizeof(arr[0]):计算数组中单个元素大小
  // 用 sizeof(arr) / sizeof(arr[0]) ,总大小 除以 单个大小 = 元素个数
  // 1. 求左下标和右下标:
  int left = 0; // 左下标
  int right = sz - 1; // 右下标
  // 因为数组是从0开始得,所以用 元素个数-1 得出 最右边下标
  int flag = 0; // 用来设置未找到情况下的处理
  //循环查找:
  while ( left <= right ) 
  // left <= right: 说明被左右下标包裹中的数组还有值,还有值就继续循环查询
  {
    // 2.确定中间元素下标:
    int mid = (left + right) / 2;
    //进行查找:
    //找到:
    if (arr[mid] == find)
    {
      printf("找到了,该值在数组中对应的下标是:%d\n", mid);
      flag = 1;
      break;
    }
    //未找到:
    //中间值小于要找的值,排除mid和小于mid左边的值      
    else if (arr[mid] < find)
    {
      left = mid + 1; // 调整左下标
    }
    //中间值大于要找的值,排除mid和大于mid右边的值
    else
    {
      right = mid - 1; // 调整右下标
    }
  }
  if (flag == 0)
  {
    printf("没找到\n");
  }
  return 0;
}


image.png

4.编写代码,演示多个字符从两端移动,向中间汇聚:

Sleep(1000)休眠1秒后再执行后面语句,Sleep需要一个windows.h的头文件

system("cls"):system函数可以执行系统命令cls是清理屏幕

//编写代码,演示多个字符从两端移动,向中间汇聚:
#include <stdio.h>
#include <string.h>
#include <windows.h>
int main()
{
  //利用字符数组来存储字符串,后面隐藏一个\0
  char arr1[] = "welcome to school!!!!!";
  char arr2[] = "######################";
  //定义左右下标:
  int left = 0;
  int right = strlen(arr1) - 1;
  // strlen(数组名称):求数组的长度,计算字符数组中\0前有多少个字符
  // 数组长度 - 1 = 最右端的小标
  while (left <= right) 
  // 如果left > right说明数组中左右下标之间已经没有值了
  // 所以有值则一直循环到无值
  {
    arr2[left] = arr1[left]; // 将arr1靠左边上的值 赋值给 arr2最左边的值
    arr2[right] = arr1[right]; // 将arr1靠右边上的值 赋值给 arr2最右边的值
    printf("%s\n", arr2); // 打印重新赋值后的arr2
    Sleep(1000); // 休眠1秒后再执行后面语句
    //Sleep需要一个windows.h的头文件
    system("cls"); // system函数可以执行系统命令,cls是清理屏幕
    left++; // ”倒数最左“
    right--; // “倒数最右”
  }
  printf("%s\n", arr2);
  return 0;
}

image.png

5.编写代码实现,模拟用户登录情景,并且只能登录三次。(只允许输入三次密码,如果密码正确则提示登录成功,如果三次均输入错误,则退出程序):

strcmp库函数:

两个字符串比较相等不能使用==(等于),应该使用strcmp库函数,需添加<string.h>头文件

       strcmp返回0表示2个字符串相等

       strcmp返回>0的数字,表示第一个字符串大于第二个字符串

       strcmp返回<0的数字,表示第一个字符串小于第二个字符串

       字符串比较大小:比较对应位置字符的ASCII值

abcd 和 abq 比较:第一位a=a,第二位b=b,第三位c<q(c的ASCII值小于q的)

       所以 abcd < abq

//5:
#include <stdio.h>
#include <string.h>
int main()
{
  int i = 0;
  //假设密码是:“123456”
  char password[20] = {0};
  int flag = 0;
  for ( i = 0; i < 3; i++) // 循环3次
  {
    printf("请输入密码:>");
    scanf("%s", password);
    // 因为password是数组,数组名本来就是地址,所以不用加取地址符&
    // 判断密码是否正确:
    // 两个字符串比较相等不能使用==(等于),
    // 应该使用strcmp库函数,需添加<string.h>头文件
    // strcmp返回0表示2个字符串相等
    // strcmp返回>0的数字,表示第一个字符串大于第二个字符串
    // strcmp返回<0的数字,表示第一个字符串小于第二个字符串
    // 字符串比较大小:比较对应位置字符的ASCII值
    // abcd 和 abq 比较:第一位a=a,第二位b=b,第三位c<q(c的ASCII值小于q的)
    // 所以 abcd < abq
    if (strcmp(password, "123456") == 0)
    {
      printf("登录成功\n");
      flag = 1;
      break; // 成功则跳出循环
    }
    else 
    {
      printf("密码错误\n");
    }
  }
  if (flag == 0) 
  {
    printf("退出程序\n");
  }
  return 0;
}

image.png

补充内容:变量的左值右值

//变量的左值和右值
#include <stdio.h>
int main()
{
  int a = 10; // 在内存申请名叫a的空间,把10存进去
  a = 20; // 把20放在a的这块空间里,这里使用的是a的空间
  int b = a; // 这里是使用了a中存放的值,把a中的20存放到b的空间(使用的是b的空间)中
  //使用变量空间的变量就是 左值  , 使用变量中存放的值的变量是 右值
  //  简单理解       (空间)                          (空间的内容) 
  return 0;
}

888332b6e66347e5bb578d4983b35b11.png

相关文章
|
14天前
|
算法 索引
【算法】——二分查找合集
二分查找基础模版和进阶模版,查找元素位置,搜索插入位置,x的平方根,山脉数组的峰顶索引,寻找峰值,点名
|
2月前
|
存储 算法 安全
2024重生之回溯数据结构与算法系列学习之串(12)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丟脸好嘛?】
数据结构与算法系列学习之串的定义和基本操作、串的储存结构、基本操作的实现、朴素模式匹配算法、KMP算法等代码举例及图解说明;【含常见的报错问题及其对应的解决方法】你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
2024重生之回溯数据结构与算法系列学习之串(12)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丟脸好嘛?】
|
2月前
|
机器学习/深度学习 人工智能 自然语言处理
【EMNLP2024】基于多轮课程学习的大语言模型蒸馏算法 TAPIR
阿里云人工智能平台 PAI 与复旦大学王鹏教授团队合作,在自然语言处理顶级会议 EMNLP 2024 上发表论文《Distilling Instruction-following Abilities of Large Language Models with Task-aware Curriculum Planning》。
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之栈和队列精题汇总(10)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第3章之IKUN和I原达人之数据结构与算法系列学习栈与队列精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2天前
|
算法 数据安全/隐私保护
室内障碍物射线追踪算法matlab模拟仿真
### 简介 本项目展示了室内障碍物射线追踪算法在无线通信中的应用。通过Matlab 2022a实现,包含完整程序运行效果(无水印),支持增加发射点和室内墙壁设置。核心代码配有详细中文注释及操作视频。该算法基于几何光学原理,模拟信号在复杂室内环境中的传播路径与强度,涵盖场景建模、射线发射、传播及接收点场强计算等步骤,为无线网络规划提供重要依据。
|
15天前
|
机器学习/深度学习 算法
基于改进遗传优化的BP神经网络金融序列预测算法matlab仿真
本项目基于改进遗传优化的BP神经网络进行金融序列预测,使用MATLAB2022A实现。通过对比BP神经网络、遗传优化BP神经网络及改进遗传优化BP神经网络,展示了三者的误差和预测曲线差异。核心程序结合遗传算法(GA)与BP神经网络,利用GA优化BP网络的初始权重和阈值,提高预测精度。GA通过选择、交叉、变异操作迭代优化,防止局部收敛,增强模型对金融市场复杂性和不确定性的适应能力。
149 80
|
3天前
|
机器学习/深度学习 数据采集 算法
基于GA遗传优化的CNN-GRU-SAM网络时间序列回归预测算法matlab仿真
本项目基于MATLAB2022a实现时间序列预测,采用CNN-GRU-SAM网络结构。卷积层提取局部特征,GRU层处理长期依赖,自注意力机制捕捉全局特征。完整代码含中文注释和操作视频,运行效果无水印展示。算法通过数据归一化、种群初始化、适应度计算、个体更新等步骤优化网络参数,最终输出预测结果。适用于金融市场、气象预报等领域。
基于GA遗传优化的CNN-GRU-SAM网络时间序列回归预测算法matlab仿真
|
3天前
|
算法
基于龙格库塔算法的锅炉单相受热管建模与matlab数值仿真
本设计基于龙格库塔算法对锅炉单相受热管进行建模与MATLAB数值仿真,简化为喷水减温器和末级过热器组合,考虑均匀传热及静态烟气处理。使用MATLAB2022A版本运行,展示自编与内置四阶龙格库塔法的精度对比及误差分析。模型涉及热传递和流体动力学原理,适用于优化锅炉效率。
|
1天前
|
移动开发 算法 计算机视觉
基于分块贝叶斯非局部均值优化(OBNLM)的图像去噪算法matlab仿真
本项目基于分块贝叶斯非局部均值优化(OBNLM)算法实现图像去噪,使用MATLAB2022A进行仿真。通过调整块大小和窗口大小等参数,研究其对去噪效果的影响。OBNLM结合了经典NLM算法与贝叶斯统计理论,利用块匹配和概率模型优化相似块的加权融合,提高去噪效率和保真度。实验展示了不同参数设置下的去噪结果,验证了算法的有效性。

热门文章

最新文章

下一篇
开通oss服务