C语言练习题解析(3)

本文涉及的产品
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 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];
}

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

目录
相关文章
|
5天前
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
52 14
|
2月前
|
程序员 编译器 数据处理
【C语言】深度解析:动态内存管理的机制与实践
【C语言】深度解析:动态内存管理的机制与实践
|
2月前
|
Serverless 编译器 C语言
【C语言】指针篇- 深度解析Sizeof和Strlen:热门面试题探究(5/5)
【C语言】指针篇- 深度解析Sizeof和Strlen:热门面试题探究(5/5)
|
4月前
|
程序员 C语言
位操作在C语言中的解析与应用
位操作在C语言中的解析与应用
95 0
|
5月前
|
存储 C语言
C语言中static关键字的作用与用法解析
C语言中static关键字的作用与用法解析
|
6月前
|
C语言
C语言实现猜数字游戏:代码详解与函数解析
C语言实现猜数字游戏:代码详解与函数解析
266 0
|
2月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
37 3
|
9天前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
34 6
|
26天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
31 6
|
2月前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
44 10

推荐镜像

更多