C语言:选择+编程(每日一练Day5)

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: C语言:选择+编程(每日一练Day5)



选择题:

题一:

1、如下程序的功能是( )
#include <stdio.h>
int main()
{
       char ch[80] = "123abcdEFG*&";
       int j;
       puts(ch);
       for(j = 0; ch[j] != '\0'; j++)
       if(ch[j] >= 'A' && ch[j] <= 'Z')
       ch[j] = ch[j] + 'e' - 'E';
       puts(ch);
       return 0;
}

A: 测字符数组ch的长度
B: 将数字字符串ch转换成十进制数
C: 将字符数组ch中的小写字母转换成大写

D: 将字符数组ch中的大写字母转换成小写

答案解析:

       一个字母对应的大写和小写之间的ASCII码值相差32,而且小写的大于大写的。所以题中'e'和'E'之间的ASCII码值相差32(ch[j]+'e'-'E'相当于ch[j]+32)。一个字母从大写转化为小写就是在它自身上+32,小写转大写则是-32。

题二:

2、对于代码段,下面描述正确的是( )
t=0;
while(printf("*"))
{
       t++;
       if (t<3)
       break;
}

A:其中循环控制表达式与0等价        B:其中循环控制表达式与'等价

C:其中循环控制表达式是不合法的    D:以上说法都不对

答案解析:

       因print(“*”)函数调用的返回值是字符串中字符的个数,即为1。所以while后面的条件恒为真,所以循环控制表达式与'0'是等价的(字符'0'不是0)。

题三:

3、以下程序运行时,若输入1abcedf2df<回车> 输出结果是( )

#include <stdio.h>
int main()
{
       char ch;
       while ((ch = getchar()) != '\n')
       {
               if (ch % 2 != 0 && (ch >= 'a' && ch <= 'z'))
               ch = ch - 'a' + 'A';
               putchar(ch);
       }
       printf("\n");
       return 0;
}

A: 1abcedf2df        B: 1ABCEDF2DF

C: 1AbCEdf2df    D: 1aBceDF2DF

答案解析:

       程序首先考虑ch的ASCII码值是不是奇数,再看是不是小写字母,同时满足时被改为大写字母。

题四:

4、下列条件语句中,功能与其他语句不同的是( )
A: if(a) printf("%d\n",x); else printf("%d\n",y);
B: if(a==0) printf("%d\n",y); else printf("%d\n",x);
C: if (a!=0) printf("%d\n",x); else printf("%d\n",y);

D: if(a==0) printf("%d\n",x); else printf("%d\n",y);

答案解析:

       D选项与众不同,其他都是a==0时输出y,a!=0时输出x。

题五:

5、我们知道C语言的 break 语句只能跳出离它最近的一层循环,可是有时候我们需要跳出多层循环,下列跳出多层循环的做法正确的是【多选】( )

A: 将程序写成函数用return结束函数,便可跳出循环;
B: 修改外层循环条件例如:

for( int i = 0 ; i < MAX1 ; i ++ )
{
    for( int j = 0 ; j < MAX2 ; j ++ )
    {
        if( condition )
    {
    i = MAX1;
    break;
}

C: 在外层循环设置判断条件例如:

for( ; symbol != 1 && condition2 ; )
{
    for( ; symbol != 1 && condition3 ; )
    {
        if( condition1 )
        symbol = 1 ;
    }
}

D: 在外层循环后面加入break例如:

for( ; condition2 ; )
{
    for( ; condition3 ; )
    {
        if( condition1 )
        symbol = 1 ;
    } 
    if(symbol == 1 )
    break ;
}

答案解析:

       此题旨在整理跳出多层循环的方法,每个选项都是正确的,代码为伪代码,condition代表逻辑表达式。

编程题:

题一:数字在升序数组中出现的次数

数字在升序数组中出现的次数_牛客题霸_牛客网 (nowcoder.com)

示例1

输入:

[1,2,3,3,3,3,4,5] , 3

复制返回值:

4

思路一:

       二分查找法:

       最直接的办法就是:二分查找一个比目标数小一位的数,由于这里是整型,所以我们直接给k+任意小数就能实现找的目标数的上界或下界,此时相减就是目标数的个数。

//函数实现二分查找
int community(int* nums,int numsLen,double k)
{
    int left = 0;
    int right = numsLen-1;
    while(left <= right)
    {
        int mid = (left+right)/2;
        if(nums[mid] > k)
        {
            right = mid-1;
        }
        else if(nums[mid] < k)
        {
            left = mid+1;
        }
    }
    return left;
}
int GetNumberOfK(int* nums, int numsLen, int k )
{
    //查找到下界下标
    int a1 = community(nums,numsLen,k+0.5);
    //查找到上界下标
    int a2 = community(nums,numsLen,k-0.5);
    return a1-a2;
}

思路二:

       采用遍历也能搞定,不过数组为非降序,采用二分查找的思想最优,先二分找到最左边的数字位置,再二分查找最右边的数字位置,两个位置相减+1就是长度了。

中间比找的值大:则要找的数字肯定在左边,right = mid - 1;
中间比找的值小:则要找的数字肯定在右边, left = mid + 1;

中间值与找的值相同:

找的最左边数字:如果mid就是left,则返回mid就行,否则重置right=mid-1,把中心不断向左偏移

找的最右边数字:如果mid就是right,则返回mid就行,否则重置left=mid+1,把中心不断向右偏移。

int get_last_or_first_idx(int* data, int len, int k, int flag) 
{ 
  //flag:0-找左边, 1-找右边
  int left = 0, right = len - 1, mid;
  while (left <= right) 
  {
    mid = left + (right - left) / 2;
    if (data[mid] > k)
      right = mid - 1;
    else if (data[mid] < k)
      left = mid + 1;
    else 
    {
      if (flag == 0) 
      { 
        //flag==0时,找最左边的数字
        if (mid == left || data[mid - 1] != k) 
          return mid;
        else 
          //把中心向左推
          right = mid - 1;
      }
      else 
      {
        //flag==1时,找最右边的数字
        if (mid == right || data[mid + 1] != k) 
          return mid;
        else 
          //把中心向右推
          left = mid + 1;
      }
    }
  } 
  return - 1;
} 
int GetNumberOfK(int* data, int dataLen, int k) 
{
  if (dataLen == 0) 
    return 0;
  int left = get_last_or_first_idx(data, dataLen, k, 0);
  int right = get_last_or_first_idx(data, dataLen, k, 1);
  if (left == -1 && right == -1) 
    return 0; //表示没有找到k这个数据
  return right - left + 1;
}

题二:整数转换

面试题 05.06. 整数转换 - 力扣(LeetCode)

思路一:

解题必懂的操作符:

      按位异或“ ^ ”: 二进制位相同为“ 0 ”,不相同为“ 1 ”;

       比如我们的数字1,二进制表示为 0001,数字2,二进制表示为 0010,对1和2进行异或,结果为 0011。

       按位“ & ”:二进制位两个都是“ 1 ”为“ 1 ”,其余都是“ 0 ”;

      右移“ >> ”:二进制位向右移动。

        首先需要先把两数异或,把值存在变量 c 中,然后再用到我们的按位与运算符,这里我们利用数字1,因为它的二进制中只有1个“ 1 ”,且在最右边,我们可以把它们进行比较。 因为1只有最右边为1,如果想32位都能比到的话,就需要我们的右移操作符,每次比较变量 c 就向右移动“ i ”位(左端补符号位),此时只需要在外面套上for循环,循环次数为32次,这样我们就能把 c 中的每一位与数字1进行按位与比较。 所以在 c 的二进制位中找到一个“ 1 ”,count++,最后程序返回count的值就行了。

int convertInteger(int A, int B)
{
    //按位异或后得到的二进制里的1的个数就是需要改变的个数
    int c = A^B;
    //记录需要改变的个数
    int count = 0;
    //遍历二进制
    for(int i = 0;i < 32;i++)
    {
        //每次向右移动i位
        if(((c >> i) & 1) == 1)
        {
            count++;
        }
    }
    return count;
}

本人实力有限可能对一些地方解释和理解的不够清晰,可以自己尝试读代码,或者评论区指出错误,望海涵!

感谢大佬们的一键三连! 感谢大佬们的一键三连! 感谢大佬们的一键三连!

                                             

目录
相关文章
|
1月前
|
NoSQL C语言 索引
十二个C语言新手编程时常犯的错误及解决方式
C语言初学者常遇错误包括语法错误、未初始化变量、数组越界、指针错误、函数声明与定义不匹配、忘记包含头文件、格式化字符串错误、忘记返回值、内存泄漏、逻辑错误、字符串未正确终止及递归无退出条件。解决方法涉及仔细检查代码、初始化变量、确保索引有效、正确使用指针与格式化字符串、包含必要头文件、使用调试工具跟踪逻辑、避免内存泄漏及确保递归有基准情况。利用调试器、编写注释及查阅资料也有助于提高编程效率。避免这些错误可使代码更稳定、高效。
216 12
|
2月前
|
Linux C语言
C语言 多进程编程(三)信号处理方式和自定义处理函数
本文详细介绍了Linux系统中进程间通信的关键机制——信号。首先解释了信号作为一种异步通知机制的特点及其主要来源,接着列举了常见的信号类型及其定义。文章进一步探讨了信号的处理流程和Linux中处理信号的方式,包括忽略信号、捕捉信号以及执行默认操作。此外,通过具体示例演示了如何创建子进程并通过信号进行控制。最后,讲解了如何通过`signal`函数自定义信号处理函数,并提供了完整的示例代码,展示了父子进程之间通过信号进行通信的过程。
|
2月前
|
Linux C语言
C语言 多进程编程(四)定时器信号和子进程退出信号
本文详细介绍了Linux系统中的定时器信号及其相关函数。首先,文章解释了`SIGALRM`信号的作用及应用场景,包括计时器、超时重试和定时任务等。接着介绍了`alarm()`函数,展示了如何设置定时器以及其局限性。随后探讨了`setitimer()`函数,比较了它与`alarm()`的不同之处,包括定时器类型、精度和支持的定时器数量等方面。最后,文章讲解了子进程退出时如何利用`SIGCHLD`信号,提供了示例代码展示如何处理子进程退出信号,避免僵尸进程问题。
|
2月前
|
消息中间件 Unix Linux
C语言 多进程编程(五)消息队列
本文介绍了Linux系统中多进程通信之消息队列的使用方法。首先通过`ftok()`函数生成消息队列的唯一ID,然后使用`msgget()`创建消息队列,并通过`msgctl()`进行操作,如删除队列。接着,通过`msgsnd()`函数发送消息到消息队列,使用`msgrcv()`函数从队列中接收消息。文章提供了详细的函数原型、参数说明及示例代码,帮助读者理解和应用消息队列进行进程间通信。
|
2月前
|
缓存 Linux C语言
C语言 多进程编程(六)共享内存
本文介绍了Linux系统下的多进程通信机制——共享内存的使用方法。首先详细讲解了如何通过`shmget()`函数创建共享内存,并提供了示例代码。接着介绍了如何利用`shmctl()`函数删除共享内存。随后,文章解释了共享内存映射的概念及其实现方法,包括使用`shmat()`函数进行映射以及使用`shmdt()`函数解除映射,并给出了相应的示例代码。最后,展示了如何在共享内存中读写数据的具体操作流程。
|
2月前
|
消息中间件 Unix Linux
C语言 多进程编程(二)管道
本文详细介绍了Linux下的进程间通信(IPC),重点讨论了管道通信机制。首先,文章概述了进程间通信的基本概念及重要性,并列举了几种常见的IPC方式。接着深入探讨了管道通信,包括无名管道(匿名管道)和有名管道(命名管道)。无名管道主要用于父子进程间的单向通信,有名管道则可用于任意进程间的通信。文中提供了丰富的示例代码,展示了如何使用`pipe()`和`mkfifo()`函数创建管道,并通过实例演示了如何利用管道进行进程间的消息传递。此外,还分析了管道的特点、优缺点以及如何通过`errno`判断管道是否存在,帮助读者更好地理解和应用管道通信技术。
|
2月前
|
Linux C语言
C语言 多进程编程(七)信号量
本文档详细介绍了进程间通信中的信号量机制。首先解释了资源竞争、临界资源和临界区的概念,并重点阐述了信号量如何解决这些问题。信号量作为一种协调共享资源访问的机制,包括互斥和同步两方面。文档还详细描述了无名信号量的初始化、等待、释放及销毁等操作,并提供了相应的 C 语言示例代码。此外,还介绍了如何创建信号量集合、初始化信号量以及信号量的操作方法。最后,通过实际示例展示了信号量在进程互斥和同步中的应用,包括如何使用信号量避免资源竞争,并实现了父子进程间的同步输出。附带的 `sem.h` 和 `sem.c` 文件提供了信号量操作的具体实现。
|
1月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
33 3
|
6天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
21 6
|
26天前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
34 10