C语言练习题解析(3)

简介: C语言练习题解析(3)

🎈前言:

  • 本专栏每篇练习将包括 5个选择题 + 2个编程题,将涵盖C语言的不同方面,包括基础语法、数据类型、控制结构、数组、指针和函数等。通过练习,你将逐步掌握C语言的基础知识和常见问题,提高你的编程技巧和解决问题的能力。
  • 我希望这个博客能够为你提供有价值的练习资源,让你在实践中不断进步。同时,我们也鼓励你在练习过程中进行思考和创新,尝试使用不同的编程方法和技巧。
  • 让我们一起挑战C语言练习题,攻克每一个难点,不断提升自己的编程技能!在评论区分享你的练习心得和问题,与我们一起交流和成长。

🌴选择题

  1. 已知函数的原型是: int fun(char b[10], int *a); ,设定义: char c[10];int d;正确的调用语句是( )

A: fun(c,&d);   B: fun(c,d);   C: fun(&c,&d);   D: fun(&c,d);

🔎正确答案:A

【解析】:

参数a是指针,要接收地址,所以BD错误。参数b可以接收的是char*,而&c的类型是char(*)[10],C错误

  1. 请问下列表达式哪些会被编译器禁止【多选】( )
int a = 248, b = 4;
int const* c = 21;
const int* d = &a;
int* const e = &b;
int const* const f = &a;

A: *c = 32;B: *d = 43C: e=&aD: f=0x321f

🔎正确答案:ABCD

【解析】:

  • A: *c = 32; - 这是错误的,因为 c 是一个指向常量整数的指针,意味着不能通过这个指针来修改所指向的值。编译器会禁止这个操作。
  • B: *d = 43; - 这是错误的,因为 d 是一个指向整数的指针,尽管它指向一个非常量整数 a,但是由于指针本身被声明为指向常量的,所以不能通过这个指针来修改所指向的值。编译器会禁止这个操作。
  • C: e = &a; - 这是错误的,因为 e 被声明为一个指向整数的常量指针,一旦指针被初始化,就不能再指向其他变量。编译器会禁止这个操作。
  • D: f = 0x321f; - 这是错误的,因为 f 被声明为一个指向常量整数的常量指针,一旦指针被初始化,就不能再指向其他变量。编译器会禁止这个操作。

【拓展】:

  • 如果 const 位于 * 的左侧,则 const 就是用来修饰指针所指向的变量,即指针指向为常量;即*c*d不能变。
  • 如果 const 位于 * 的右侧,则 const 就是修饰指针本身,即指针本身是常量;即ef不能变。
  1. 以下程序的输出结果为( )
#include <stdio.h>
int i;
void prt()
{
  for (i = 5; i < 8; i++)
    printf("%c", '*');
  printf("\t");
}
int main()
{
  for (i = 5; i <= 8; i++)
    prt();
  return 0;
}

A: ***   B: *** *** *** ***   C: *** ***   D: * * *

🔎正确答案:A

【解析】:

全局变量i,在main()中修改为5,第一次在prt()中执行循环输出三次'*'i被修改为8,回到main()中第二次调用prt()时,i<8为假,循环结束没输出,执行一次print("\t"),再次回到主函数后i++变为9i<=8为假,循环结束。

  1. 下面代码段的输出是( )
#include<stdio.h>
int main()
{
  int a = 3;
  printf("%d\n", (a += a -= a * a));
  return 0;
}

A: -6   B: 12   C: 0   D: -12

🔎正确答案:D

【解析】:

a+=a-=a*a等价于a=a+(a=a-a*a),即先计算a=a-a*a,所以此时a的值为3-3*3=-6,再计算-6+(-6)=-12赋值给a,所以a的值为-12,也就是整个表达式的值,所以应选择D

  1. 下列不能实现死循环的是( )

A: while(1){}   B: for(;1;){}   C: do{}while(1);   D: for(;0;){}

🔎正确答案:D

【解析】:

只有条件为真时才进行循环,ABC中1为真,D中0为假


🌴编程题

📌记负均正

【牛客网链接:HJ97 记负均正

【题目信息】:

【答案解析】:

  • 这道题其实通过 scanf 捕捉数据即可,统计负数个数,以及正数个数,并且在统计正数个数的过程中求取正数总和,最后计算得出平均数即可。需要注意的是所有数字中0是不统计在内的。
#include <stdio.h>
int main()
{
  int n;
  while (~scanf("%d", &n))//多组输入 (也可以写成 scanf("%d",&n)!=EOF )
  {
    int count1 = 0, count2 = 0, tmp;
    double sum = 0;
    for (int i = 0; i < n; i++) 
    {
      scanf("%d", &tmp);//对每次输入的整数进行判断
      if (tmp < 0) 
      {
        count1++; //统计负数个数
      }
      else if (tmp > 0) 
      {
        sum += tmp; //正数求和
        count2++; //统计大于0的正数个数,这样是因为题目说明0不算在内
      }
    }
    printf("%d ", count1);
        if (count2==0)//如果没有正数,则平均值为0 
        {
            printf("%.1lf\n",0.0);
        }
        else//若有正数,则平均值为所有正数的和除以正数的个数
        {
            printf("%.1lf\n",sum / count2);
        }
  } 
  return 0;
}

📌旋转数组的最小数字

【牛客网链接:JZ11 旋转数组的最小数字

【题目信息】:

【答案解析】:

  • 暴力破解:遍历数组找出最小值即可(时间复杂度:O(N)
//暴力破解:遍历数组找出最小值即可
int minNumberInRotateArray(int* nums, int numsLen) {
  int i = 0;
  for (i = 0; i < numsLen - 1; i++)
  {
    if (nums[i] > nums[i + 1])//只要出现降序则说明出现了最小值,比如[4,5,1,2,3] :5到1是降序,则1就是最小值;
      return nums[i + 1];
  }
  return nums[0];//如果没出现降序,说明没旋转,则第一个数就是最小值。
}
  • 更优思想:采用二分查找,这个题主要分析三种旋转情况[1, 2, 3, 4, 5],使用中间值与右端进行比较。(时间复杂度:O(logN)
  1. 中间大于右边 [3, 4, 5, 1, 2],这种情况下,最小数一定在右边;则left = middle + 1
  2. 中间等于右边 [1, 0, 1, 1, 1], 这个是[0, 1, 1, 1, 1] 旋转过来的,这时候需要缩小范围 right--;,注意不能是left++,因为是非降序数组,所以要缩小右边范围,把较小值向右推,符合我们的判断规则。
  3. 中间小于右边 [5, 1, 2, 3, 4], 这种情况下,最小数字则在左半边;则right = middle
//更优思想:采用二分查找
int minNumberInRotateArray(int* nums, int numsLen) {
  int left = 0;
  int right = numsLen - 1;
  while (left < right)
  {
    int mid = left + (right - left) / 2;//避免整数溢出
    if (nums[mid] > nums[right])
    {
      left = mid + 1;//最小值在[left+1,right]
    }
    else if (nums[mid] == nums[right])
    {
      right--;//缩短数组,将最小值范围右移
    }
    else
    {
      right = mid;//最小值在[left,mid]
    }
  }
  return nums[left];
}

🔥今天的内容就到这里了,有什么问题的话欢迎大家在评论区讨论,也可以私信博主为你解答,如果觉得博主的文章还不错的话, 请👍三连支持一下博主哦🤞

目录
相关文章
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
989 14
|
存储 编译器 C语言
【C语言】数据类型全解析:编程效率提升的秘诀
在C语言中,合理选择和使用数据类型是编程的关键。通过深入理解基本数据类型和派生数据类型,掌握类型限定符和扩展技巧,可以编写出高效、稳定、可维护的代码。无论是在普通应用还是嵌入式系统中,数据类型的合理使用都能显著提升程序的性能和可靠性。
567 8
|
存储 算法 C语言
【C语言】深入浅出:C语言链表的全面解析
链表是一种重要的基础数据结构,适用于频繁的插入和删除操作。通过本篇详细讲解了单链表、双向链表和循环链表的概念和实现,以及各类常用操作的示例代码。掌握链表的使用对于理解更复杂的数据结构和算法具有重要意义。
3588 6
|
存储 网络协议 算法
【C语言】进制转换无难事:二进制、十进制、八进制与十六进制的全解析与实例
进制转换是计算机编程中常见的操作。在C语言中,了解如何在不同进制之间转换数据对于处理和显示数据非常重要。本文将详细介绍如何在二进制、十进制、八进制和十六进制之间进行转换。
2044 5
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
330 5
|
3月前
|
存储 C语言
`scanf`是C语言中用于按格式读取标准输入的函数
`scanf`是C语言中用于按格式读取标准输入的函数,通过格式字符串解析输入并存入指定变量。需注意输入格式严格匹配,并建议检查返回值以确保读取成功,提升程序健壮性。
1048 0
|
11月前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
714 23
|
5月前
|
安全 C语言
C语言中的字符、字符串及内存操作函数详细讲解
通过这些函数的正确使用,可以有效管理字符串和内存操作,它们是C语言编程中不可或缺的工具。
338 15
|
10月前
|
人工智能 Java 程序员
一文彻底搞清楚C语言的函数
本文介绍C语言函数:函数是程序模块化的工具,由函数头和函数体组成,涵盖定义、调用、参数传递及声明等内容。值传递确保实参不受影响,函数声明增强代码可读性。君志所向,一往无前!
442 1
一文彻底搞清楚C语言的函数
|
11月前
|
算法 C语言
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
本文档介绍了如何编写两个子函数,分别求任意两个整数的最大公约数和最小公倍数。内容涵盖循环控制与跳转语句的使用、最大公约数的求法(包括辗转相除法和更相减损术),以及基于最大公约数求最小公倍数的方法。通过示例代码和测试说明,帮助读者理解和实现相关算法。最终提供了完整的通关代码及测试结果,确保编程任务的成功完成。
630 15
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】

推荐镜像

更多
  • DNS