• 第五站:操作符(终幕)(一些经典的题目)


    目录

    一、分析下面的代码

    二、统计二进制中1的个数

    解一:(求出每一个二进制位,来统计1的个数)

    解二:(利用左我们移或右移操作符和按位与)

    解三:(效率最高的解法,利用n=n&(n-1)求解)

    举一反三:

    三、打印整数二进制的奇数位和偶数位

    解一:(暴力求解法)

    解二:(利用位操作符)

     四、求两个数二进制中不同位的个数

    解一:(利用移位操作符和按位与)

     解二:(异或操作符求解)

    五、 获取月份天数

    解一:(采用switch语句)

     解二:运用一个数组来存放天数

     六、一个经典的必错题!(算术转换)

    七、序列中删除指定数字

    解一:两层遍历,后面元素往前覆盖的解法

    解二: 利用两个下标来求解

    总结


    一、分析下面的代码

    分析下面的代码,并给出运行结果。

    1. #include <stdio.h>
    2. int main()
    3. {
    4. int a, b, c;
    5. a = 5;
    6. c = ++a;
    7. b = ++c, c++, ++a, a++;
    8. b += a++ + c;
    9. printf("a = %d b = %d c = %d\n:", a, b, c);
    10. return 0;
    11. }

    解析:这道题其实就是考察优先级的问题。在这里需要注意的是逗号的优先级最低,因此在第七行代码中,是先将c赋给了b,然后才继续往下的逗号运算的。这里很容易出现错误。

    二、统计二进制中1的个数

    题目描述:输入一个整数 n ,输出该数32位二进制表示中1的个数。其中负数用补码表示。

    链接:二进制中1的个数__牛客网
    来源:牛客网

    解一:(求出每一个二进制位,来统计1的个数)

    我们知道求一个数的二进制是采用取模,和除法的运算来实现的。所以我们可以利用这个思想来进行求解。定义一个计数器count,求出每一个二进制位,如果等于1,则count++

    代码如下

    1. #include<stdio.h>
    2. int Numberof1(int n)
    3. {
    4. int count = 0;
    5. while (n)
    6. {
    7. if (n % 2 == 1)
    8. {
    9. count++;
    10. }
    11. n = n / 2;
    12. }
    13. return count;
    14. }
    15. int main()
    16. {
    17. int n = 0;
    18. scanf("%d", &n);
    19. int ret = Numberof1(n);
    20. printf("%d", ret);
    21. return 0;
    22. }

    我们测试后,发现对于正数,都是没有问题的,但是如果输入负数的话,就会出现问题。如下图所示

     

     因此我们需要进行优化代码,其实题目要求说,负数用补码来代替,但是他这里的运算是直接对原码翻译成的十进制数进行运算的。而数据在内存中存储的是补码,所以其实我们完全可以使用一个unsigned来进行修饰。这样就把我们的补码当作原码来进行运算了。

    代码如下所示

    1. #include<stdio.h>
    2. int Numberof1(unsigned int n)
    3. {
    4. int count = 0;
    5. while (n)
    6. {
    7. if (n % 2 == 1)
    8. {
    9. count++;
    10. }
    11. n = n / 2;
    12. }
    13. return count;
    14. }
    15. int main()
    16. {
    17. int n = 0;
    18. scanf("%d", &n);
    19. int ret = Numberof1(n);
    20. printf("%d", ret);
    21. return 0;
    22. }

    运行结果如下图所示

     虽然这种解法经过我们测试后是没有问题的,但是这种解法在我们的牛客网平台其实是过不去的,因为我们这个已经改了函数的参数类型了,而题目要求是不能修改,所以我们还需要另寻他法。

    解二:(利用左我们移或右移操作符和按位与)

    其实我们要注意到,这道题目是针对二进制的,二进制在计算机内部是补码的形式。我们的第一种方式还需要将补码弄成原码,才成功做出来题。这种方法虽然可以,但是我们其实没有必要这么麻烦。我们可以直接对补码进行操作。而对补码进行操作的操作符就是我们的位操作符了,位操作符一共有&,|,^,>>,<<这五种

    我们观察这五种操作符,我们可以想到,按位与&有一个比较良好的性质,任何一个二进制位按位与1结果仍然为该二进制位。因此我们可以思考让这个数的每一位都按位与1。如果结果为1,则count++。那么如果让每一位都能按位与1呢?其实我们可以使用移位操作符,这里我们既可以左移1,也可以右移该二进制序列。两种方法皆可

    代码如下

    1. #include<stdio.h>
    2. int Numberof1(int n)
    3. {
    4. int count = 0;
    5. int i = 0;
    6. for (i = 0; i < 32; i++)
    7. {
    8. if (((n >> i) & 1 )== 1)
    9. {
    10. count++;
    11. }
    12. }
    13. return count;
    14. }
    15. int main()
    16. {
    17. int n = 0;
    18. scanf("%d", &n);
    19. int ret = Numberof1(n);
    20. printf("%d", ret);
    21. return 0;
    22. }

     而我们将这个代码的函数部分放到牛客网上运行,可以显示运行通过

     当然除此以外,其实还可以采用按位或的性质,任何一个二进制位按位或0,结果仍然为该二进制数。这种方法与上述方法完全是对偶的,因此不在赘述。

    解三:(效率最高的解法,利用n=n&(n-1)求解)

    解二中,方法虽然可以实现,但是效率太低,因为无论是多大的数,都需要执行32次循环才可以解决问题。因此我们考虑使用这个公式,为什么是要利用这个公式呢?我们举一个例子,模拟一下这个公式的执行

     我们发现,其实每执行一次,相当于n少了一个1.因此我们可以每执行一次,count++就会实现我们的目标。其实从我们数学的角度来思考,n每减1,然后按位与n所得到的结果就会使得我们原来的n少一个1

    代码如下所示

    1. #include<stdio.h>
    2. int Numberof1(int n)
    3. {
    4. int count = 0;
    5. while (n)
    6. {
    7. n = n & (n - 1);
    8. count++;
    9. }
    10. return count;
    11. }
    12. int main()
    13. {
    14. int n = 0;
    15. scanf("%d", &n);
    16. int ret = Numberof1(n);
    17. printf("%d", ret);
    18. return 0;
    19. }

    牛客网上运行结果为

    举一反三:

    关于这道题,我们其实还可以引发一些思考,我们看一下这道题

    写出一个代码,判断某一个数是否是2的次方

    这个题目当然可以用暴力破解。但是我们其实有效率更高的方式,因为某一个数是2的次方,那就说明,该数的二进制序列中只有一个1。由此题目豁然开朗、迎刃而解。我们上面的三种方式可以完成这个判断

    利用解三完成的代码如下

    1. #include<stdio.h>
    2. int main()
    3. {
    4. int n = 0;
    5. scanf("%d", &n);
    6. int flag = 0;
    7. if ((n & (n - 1)) == 0)
    8. {
    9. flag = 1;
    10. }
    11. if (flag == 1)
    12. {
    13. printf("是2的次方");
    14. }
    15. else
    16. {
    17. printf("不是2的次方");
    18. }
    19. return 0;
    20. }

    三、打印整数二进制的奇数位和偶数位

    题目描述:输入一个数,打印出他二进制的奇数位和偶数位

    解一:(暴力求解法)

    对于二进制的题目,思考最少,最简单,最暴力的方法就是求出每一位,然后进行打印出奇数位和偶数位,值得注意的是,数据在内存中存储的是二进制位,因此会出现负数的问题,因此我们需要使用unsigned进行修饰。来避免一些问题

    代码如下

    1. #include<stdio.h>
    2. int main()
    3. {
    4. unsigned n = -1;
    5. int i = 0;
    6. unsigned int tmp = 0;
    7. //获取奇数位
    8. tmp = n;
    9. for (i = 1; i <= 32; i = i + 2)
    10. {
    11. printf("从右往左第%d位: %u\n",i, tmp % 2);
    12. tmp = tmp / 2;
    13. tmp = tmp / 2;
    14. }
    15. //获取偶数位
    16. tmp = n;
    17. for (i = 2; i <= 32; i = i + 2)
    18. {
    19. tmp = tmp / 2;
    20. printf("从右往左第%d位: %u\n", i, tmp % 2);
    21. tmp = tmp / 2;
    22. }
    23. return 0;
    24. }

    解二:(利用位操作符)

    第一种方法实在太暴力,效率也很低。既然是跟二进制相关,那肯定要考虑位操作符相关的方法。有了前面第二题的第二中方法的参考,我们也可以按照移位操作符和按位与来实现一下

    代码如下:

    1. #include<stdio.h>
    2. void Print(int n)
    3. {
    4. int i = 0;
    5. //打印奇数
    6. printf("打印奇数:");
    7. for (i = 30; i >= 0; i = i - 2)
    8. {
    9. printf("%d ", (n >> i) & 1);
    10. }
    11. printf("\n");
    12. //打印偶数
    13. printf("打印偶数:");
    14. for (i = 31; i >= 0; i = i - 2)
    15. {
    16. printf("%d ", (n >> i) & 1);
    17. }
    18. printf("\n");
    19. }
    20. int main()
    21. {
    22. int n = 0;
    23. scanf("%d", &n);
    24. Print(n);
    25. }

    运行结果为

     四、求两个数二进制中不同位的个数

    链接:两个整数二进制位不同个数__牛客网
    来源:牛客网

    输入两个整数,求两个整数二进制格式有多少个位不同

    解一:(利用移位操作符和按位与)

    这种思想我们在第二题,第三题均已经提到过。我们继续来利用这种思想来完成这道题

    代码如下:

    1. #include<stdio.h>
    2. int main()
    3. {
    4. int n = 0, m = 0;
    5. int count = 0;
    6. scanf("%d %d", &m, &n);
    7. int i = 0;
    8. for (i = 0; i <= 32; i++)
    9. {
    10. if (((n >> i) & 1) != ((m >> i) & 1))
    11. {
    12. count++;
    13. }
    14. }
    15. printf("%d", count);
    16. return 0;
    17. }

     解二:(异或操作符求解)

    上面的运行后效率是比较低的,那么有没有比较高的效率呢?我们说是有的,我们将两个数异或,能得到一共全新的二进制序列,对于这个二进制序列,我们就只需要统计这个二进制序列中1的个数,即可得到答案,而统计二进制序列中1的个数,这正好就是第二个题

    代码如下:

    1. #include<stdio.h>
    2. int get_dif_count(int n)
    3. {
    4. int count = 0;
    5. while (n)
    6. {
    7. n = n & (n - 1);
    8. count++;
    9. }
    10. return count;
    11. }
    12. int main()
    13. {
    14. int a = 0;
    15. int b = 0;
    16. scanf("%d %d", &a, &b);
    17. int c = a ^ b;
    18. int n = get_dif_count(c);
    19. printf("%d", n);
    20. return 0;
    21. }

    五、 获取月份天数

        获取月份天数链接:获取月份天数_牛客网

       来源:牛客网

    解一:(采用switch语句)

    对于这种,题目我们可以直接采取switch语句,唯一需要注意的是2月需要判断闰年

    代码如下:

    1. #include<stdio.h>
    2. int is_leap_year(int y)
    3. {
    4. if ((y % 4 == 0) && (y % 100 != 0) || (y % 400) == 0)
    5. {
    6. return 1;
    7. }
    8. return 0;
    9. }
    10. int get_day_of_month(int y, int m)
    11. {
    12. int day = 0;
    13. switch (m)
    14. {
    15. case 1:
    16. case 3:
    17. case 5:
    18. case 7:
    19. case 8:
    20. case 10:
    21. case 12:
    22. day = 31;
    23. break;
    24. case 4:
    25. case 6:
    26. case 9:
    27. case 11:
    28. day = 30;
    29. break;
    30. case 2:
    31. day = 28;
    32. if (is_leap_year(y) == 1)
    33. {
    34. day++;
    35. }
    36. }
    37. return day;
    38. }
    39. int main()
    40. {
    41. int y = 0, m = 0;
    42. while (scanf("%d %d", &y, &m) == 2)
    43. {
    44. int ret = get_day_of_month(y, m);
    45. printf("%d\n", ret);
    46. }
    47. return 0;
    48. }

     解二:运用一个数组来存放天数

    上面switch语句写法固然可以,但显得太过于啰嗦,我们使用一个数组,可以使代码更简洁

    1. #include <stdio.h>
    2. int is_leap_year(int y)
    3. {
    4. if ((y % 4 == 0) && (y % 100 != 0) || (y % 400) == 0)
    5. {
    6. return 1;
    7. }
    8. return 0;
    9. }
    10. int main() {
    11. int y, m;
    12. while (scanf("%d %d", &y, &m) == 2)
    13. {
    14. int day[] = { 31,28,31,30,31,30,31,31,30,31,30,31 };
    15. if (is_leap_year(y) == 1)
    16. {
    17. day[1]++;
    18. }
    19. printf("%d\n", day[m - 1]);
    20. }
    21. return 0;
    22. }

     六、一个经典的必错题!(算术转换)

    如下所示,阅读代码,并给出答案。

    1. #include
    2. int i;
    3. int main()
    4. {
    5. i--;
    6. if (i > sizeof(i))
    7. {
    8. printf(">\n");
    9. }
    10. else
    11. {
    12. printf("<\n");
    13. }
    14. return 0;
    15. }

    很多人一看这道题,觉得太简单了,直接就是-1<4,选<,这样做就进入了陷阱了。

    我们先看一下运行结果,为>

     那么这是为什么呢?其实这是因为中间发生了算术转换。i是一个全局变量,默认是0,然后i--,此时i变为了-1。但是sizeof计算的结果类型是size_t,而是size_t就是unsigned int。-1是一个普通整型,按照我们之前所说的要根据层级,向上发生算术转换,因此-1变为了unsigned int类型,而-1的补码是11111111 11111111 11111111 11111111,他变为unsigned int类型后,原码就是补码了,导致原码很大。因此-1就被隐式转换成了一个超级大的数,他当然大于4了。因此结果就是>

    七、序列中删除指定数字

    链接:序列中删除指定数字_牛客网

    来源:牛客网

    解一:两层遍历,后面元素往前覆盖的解法

    这个方法算是一个暴力的解法。我们直接去遍历这个数组,遇到删除的元素,将后面的整体往前挪一个单位,但是这样的话要注意某一个位置出现连续重复数字,为了避免这种情况,我们挪之后,还需要重新检验一下当前的i是否需要删除

    代码如下:

    1. #include <stdio.h>
    2. int main() {
    3. int n=0;
    4. scanf("%d",&n);
    5. int arr[n];
    6. int i=0;
    7. for(i=0;i<n;i++)
    8. {
    9. scanf("%d",&arr[i]);
    10. }
    11. int num=0;
    12. int tmp=n;
    13. scanf("%d",&num);
    14. for(i=0;i<tmp;i++)
    15. {
    16. if(arr[i]==num)
    17. {
    18. int j=0;
    19. tmp=tmp-1;
    20. for(j=i;j<tmp;j++)
    21. {
    22. arr[j]=arr[j+1];
    23. }
    24. i--;
    25. }
    26. }
    27. for(i=0;i<tmp;i++)
    28. {
    29. printf("%d ",arr[i]);
    30. }
    31. return 0;
    32. }

    解二: 利用两个下标来求解

    我们可以这样想,我们先遍历我们的数组,然后定义两个变量i,j作为下标,我们可以通过这两个下标,当数组元素与待删除数字不相同时候,我们将i下标的数组元素赋给j下标的数组元素。然后j++,i++。当数组元素与待删除数字相同时,我们直接跳过该元素,只让i++,j不变。最后我们遍历数组的时候,j也刚好是我们数组的需要的大小。

    代码如下:

    1. #include <stdio.h>
    2. int main() {
    3. int n=0;
    4. scanf("%d",&n);
    5. int arr[n];
    6. int i=0;
    7. for(i=0;i<n;i++)
    8. {
    9. scanf("%d",&arr[i]);
    10. }
    11. int j=0;
    12. int del=0;
    13. scanf("%d",&del);
    14. for(i=0;i<n;i++)
    15. {
    16. if(arr[i]!=del)
    17. {
    18. arr[j++]=arr[i];
    19. }
    20. }
    21. for(i=0;i<j;i++)
    22. {
    23. printf("%d ",arr[i]);
    24. }
    25. return 0;
    26. }

    运行结果为


    总结

    本小节主要讲解了一些经典的题目,内容比较丰富,希望对大家有所帮助。不要忘记点赞+收藏哦!!!

  • 相关阅读:
    CAS与自旋锁、ABA问题
    用Prophet在Python中进行时间序列预测
    生物神经网络 原理分析研读01
    一周面了 20 多场,新鲜面经奉上
    【2023年11月第四版教材】第18章《项目绩效域》(第二部分)
    营收下滑,腾讯游戏还能保持「王者」地位吗?
    ChatGPT 基本用法!ChatGPT4的prompt的使用例子!
    【ASP.NET Core】应用脱机文件 (app_offline.htm)
    Python标准库之pickle
    Xilinx FPGA平台DDR3设计详解(一):DDR SDRAM系统框架
  • 原文地址:https://blog.csdn.net/jhdhdhehej/article/details/128047059