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

news/2024/4/27 5:14:00/文章来源:https://blog.csdn.net/jhdhdhehej/article/details/128047059

目录

一、分析下面的代码

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

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

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

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

举一反三:

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

解一:(暴力求解法)

解二:(利用位操作符)

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

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

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

五、 获取月份天数

解一:(采用switch语句)

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

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

七、序列中删除指定数字

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

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

总结


一、分析下面的代码

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

#include <stdio.h>
int main()
{int a, b, c;a = 5;c = ++a;b = ++c, c++, ++a, a++;b += a++ + c;printf("a = %d b = %d c = %d\n:", a, b, c);return 0;
}

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

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

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

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

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

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

代码如下

#include<stdio.h>
int Numberof1(int n)
{int count = 0;while (n){if (n % 2 == 1){count++;}n = n / 2;}return count;
}
int main()
{int n = 0;scanf("%d", &n);int ret = Numberof1(n);printf("%d", ret);return 0;
}

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

 

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

代码如下所示

#include<stdio.h>
int Numberof1(unsigned int n)
{int count = 0;while (n){if (n % 2 == 1){count++;}n = n / 2;}return count;
}
int main()
{int n = 0;scanf("%d", &n);int ret = Numberof1(n);printf("%d", ret);return 0;
}

运行结果如下图所示

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

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

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

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

代码如下

#include<stdio.h>
int Numberof1(int n)
{int count = 0;int i = 0;for (i = 0; i < 32; i++){if (((n >> i) & 1 )== 1){count++;}}return count;
}
int main()
{int n = 0;scanf("%d", &n);int ret = Numberof1(n);printf("%d", ret);return 0;
}

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

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

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

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

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

代码如下所示

#include<stdio.h>
int Numberof1(int n)
{int count = 0;while (n){n = n & (n - 1);count++;}return count;
}
int main()
{int n = 0;scanf("%d", &n);int ret = Numberof1(n);printf("%d", ret);return 0;
}

牛客网上运行结果为

举一反三:

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

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

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

利用解三完成的代码如下

#include<stdio.h>
int main()
{int n = 0;scanf("%d", &n);int flag = 0;if ((n & (n - 1)) == 0){flag = 1;}if (flag == 1){printf("是2的次方");}else{printf("不是2的次方");}return 0;
}

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

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

解一:(暴力求解法)

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

代码如下

#include<stdio.h>
int main()
{unsigned n = -1;int i = 0;unsigned int tmp = 0;//获取奇数位tmp = n;for (i = 1; i <= 32; i = i + 2){printf("从右往左第%d位: %u\n",i, tmp % 2);tmp = tmp / 2;tmp = tmp / 2;}//获取偶数位tmp = n;for (i = 2; i <= 32; i = i + 2){tmp = tmp / 2;printf("从右往左第%d位: %u\n", i, tmp % 2);tmp = tmp / 2;}return 0;
}

解二:(利用位操作符)

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

代码如下:

#include<stdio.h>
void Print(int n)
{int i = 0;//打印奇数printf("打印奇数:");for (i = 30; i >= 0; i = i - 2){printf("%d ", (n >> i) & 1);}printf("\n");//打印偶数printf("打印偶数:");for (i = 31; i >= 0; i = i - 2){printf("%d ", (n >> i) & 1);}printf("\n");
}
int main()
{int n = 0;scanf("%d", &n);Print(n);
}

运行结果为

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

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

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

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

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

代码如下:

#include<stdio.h>
int main()
{int n = 0, m = 0;int count = 0;scanf("%d %d", &m, &n);int i = 0;for (i = 0; i <= 32; i++){if (((n >> i) & 1) != ((m >> i) & 1)){count++;}}printf("%d", count);return 0;
}

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

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

代码如下:

#include<stdio.h>
int get_dif_count(int n)
{int count = 0;while (n){n = n & (n - 1);count++;}return count;
}
int main()
{int a = 0;int b = 0;scanf("%d %d", &a, &b);int c = a ^ b;int n = get_dif_count(c);printf("%d", n);return 0;
}

五、 获取月份天数

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

   来源:牛客网

解一:(采用switch语句)

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

代码如下:

#include<stdio.h>
int is_leap_year(int y)
{if ((y % 4 == 0) && (y % 100 != 0) || (y % 400) == 0){return 1;}return 0;
}
int get_day_of_month(int y, int m)
{int day = 0;switch (m){case 1:case 3:case 5:case 7:case 8:case 10:case 12:day = 31;break;case 4:case 6:case 9:case 11:day = 30;break;case 2:day = 28;if (is_leap_year(y) == 1){day++;}}return day;
}
int main()
{int y = 0, m = 0;while (scanf("%d %d", &y, &m) == 2){int ret = get_day_of_month(y, m);printf("%d\n", ret);}return 0;
}

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

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

#include <stdio.h>
int is_leap_year(int y)
{if ((y % 4 == 0) && (y % 100 != 0) || (y % 400) == 0){return 1;}return 0;
}
int main() {int y, m;while (scanf("%d %d", &y, &m) == 2){int day[] = { 31,28,31,30,31,30,31,31,30,31,30,31 };if (is_leap_year(y) == 1){day[1]++;}printf("%d\n", day[m - 1]);}return 0;
}

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

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

#include <stdio.h>
int i;
int main()
{i--;if (i > sizeof(i)){printf(">\n");}else{printf("<\n");}return 0;
}

很多人一看这道题,觉得太简单了,直接就是-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是否需要删除

代码如下:

#include <stdio.h>int main() {int n=0;scanf("%d",&n);int arr[n];int i=0;for(i=0;i<n;i++){scanf("%d",&arr[i]);}int num=0;int tmp=n;scanf("%d",&num);for(i=0;i<tmp;i++){if(arr[i]==num){int j=0;tmp=tmp-1;for(j=i;j<tmp;j++){arr[j]=arr[j+1];}i--;}}for(i=0;i<tmp;i++){printf("%d ",arr[i]);}return 0;
}

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

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

代码如下:

#include <stdio.h>int main() {int n=0;scanf("%d",&n);int arr[n];int i=0;for(i=0;i<n;i++){scanf("%d",&arr[i]);}int j=0;int del=0;scanf("%d",&del);for(i=0;i<n;i++){if(arr[i]!=del){arr[j++]=arr[i];}}for(i=0;i<j;i++){printf("%d ",arr[i]);}return 0;
}

运行结果为

 


总结

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_39014.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【iOS】—— GET和POST以及AFNetworking框架

GET和POST以及AFNetworking框架 文章目录GET和POST以及AFNetworking框架GET和POSTGET和POST区别GETGET请求步骤GET请求代码POSTPOST请求步骤POST请求代码AFNetworking简介添加头文件GETGET方法GET方法参数GET方法代码样例POSTPOST方法第一种&#xff1a;第二种&#xff1a;先来…

C++:STL之Vector实现

vector各函数 #include<iostream> #include<vector> using namespace std;namespace lz {//模拟实现vectortemplate<class T>class vector{public:typedef T* iterator;typedef const T* const_iterator;//默认成员函数vector(); …

Netty进阶——粘包与半包(代码示例)

目录一、消息粘包和消息半包的概述1.1、消息粘包1.2、消息半包二、粘包现象代码示例2.1、粘包现象服务端示例代码2.2、粘包现象客户端示例代码2.3、分别启动服务端&#xff0c;客户端&#xff0c;查看服务端结果输出三、半包现象代码示例3.1、半包现象服务端示例代码3.2、半包现…

【强化学习论文】小样本策略泛化的提示决策转换器

文献题目&#xff1a;Prompting Decision Transformer for Few-Shot Policy Generalization 摘要 人类可以利用先前的经验并从少量演示中学习新任务。与旨在通过更好的算法设计实现快速适应的离线元强化学习相比&#xff0c;我们研究了架构归纳偏差对少样本学习能力的影响。我…

不懂单链表? 这篇文章就帮你搞明白

坚持看完&#xff0c;结尾有思维导图总结 链表对指针的操作要求不低链表的概念链表的特性链表的功能(最重要)定义和初始化头插头删细节说明尾插尾删寻找链表元素与打印链表在 某位置后插入删除在某位置的插入删除销毁链表链表的概念 什么是链表 官方概念&#xff1a;链表是一种…

显卡---显卡驱动---CUDA---Cudnn

1. 背景 最近在follow百度的CAE这篇论文时&#xff0c;源码需要的环境为&#xff1a; python 3.7 cuda: 11.0 cudnn: 8.0.4 gcc 8.2 该版本要求与我目前使用的服务器上的CUDA版本不相符合。因此搜索了一篇国外小哥的文章&#xff0c;讲述了如何在一台服务器上安装多个CUDA和Cud…

Node之Express学习笔记

一.Express 1.1什么是Express Express的作用和Node.js内置的http模块类似 专门用来创建Web服务器的 本质&#xff1a;npm上的第三方包&#xff0c;提供快速创建Web服务器的便捷方法 1.2Express能做什么 Web网站服务器&#xff1a;专门提供Web网页资源的服务器 API接口服务器&…

FPGA----ZCU106基于axi-hp通道的pl与ps数据交互(全网唯一最详)

1、大家好&#xff0c;今天给大家带来的内容是&#xff0c;基于AXI4协议的采用AXI-HP通道完成PL侧数据发送至PS侧&#xff08;PS侧数据发送至PL侧并没有实现&#xff0c;但是保留了PL读取PS测数据的接口&#xff09; 2、如果大家用到SoC这种高级功能&#xff0c;那大家应该对于…

Linux进阶-进程间通信(ipc)

进程间通信&#xff1a;数据传输、资源共享、事件通知、进程控制。 Linux系统下的ipc 早期unix系统 ipc&#xff1a;管道&#xff08;数据传输&#xff09;、信号&#xff08;事件通知&#xff09;、fifo&#xff08;数据传输&#xff09;。 system-v ipc&#xff08;贝尔实…

Kubernetes之Pod初始化容器

Kubernetes之Pod初始化容器 概述 ​ 初始化是很多编程语言普遍关注的问题&#xff0c;甚至有些编程语言直接支持模式构造来生成初始化程序&#xff0c;这些用于进行初始化的程序结构称为初始化器或初始化列表。初始化代码要首先运行&#xff0c;且只能运行一次&#xff0c;它们…

CAPM资产定价模型

本文是Quantitative Methods and Analysis: Pairs Trading此书的读书笔记。 CAPM(Capital Asset Pricing Model)资产定价模型 这个模型在金融界的影响就是beta这个词的使用。CAPM模型用两个部分的回报之和来解释一个资产的回报。其中一个部分是指市场&#xff08;称为market)…

时间序列:时间序列模型---自回归过程(AutoRegressive Process)

本文是Quantitative Methods and Analysis: Pairs Trading此书的读书笔记。 这次我们构造一个由无限的白噪声实现&#xff08;white noise realization) 组成的时间序列&#xff0c;即。这个由无限数目的项组成的值却是一个有限的值&#xff0c;比如时刻的值为&#xff0c; 而…

Magic Leap 2设计和开发幕后花絮

Magic Leap今年发布新款AR头显Magic Leap 2&#xff0c;相比于上一代Magic Leap 1&#xff0c;新品更专注于B端场景&#xff0c;自公布以来&#xff0c;Magic Leap不仅对公司策略、理念更加透明&#xff0c;也不断公开ML2产品设计背后的思考。相比于ML1&#xff0c;ML2的设计有…

粒子群算法查找函数最小值

实现 函数 F01、F06 的优化测试 以下内容是现有算法的运行结果、调参分析、及代码实现&#xff0c;用于给其他人提供参考&#xff0c;懒得改了hh 1. 运行结果 参数 w 0.5 &#xff08;可更改&#xff09; c1 2.0 &#xff08;可更改&#xff09; c2 2.0 &#xff08;可更改&…

Echart 柱状图,X轴斜着展示

option { color: [‘#3398DB’], tooltip: { trigger: ‘axis’, axisPointer: { // 坐标轴指示器&#xff0c;坐标轴触发有效 type: ‘shadow’ // 默认为直线&#xff0c;可选为&#xff1a;‘line’ | ‘shadow’ } }, grid: { left: ‘3%’, right: ‘4%’, bottom: ‘3%’…

iPhone升级iOS 16后出现提示“面容ID不可用”怎么办?

最近&#xff0c;很多用户在苹果社区反馈&#xff0c;iPhone升级iOS 16后Face ID不能用了&#xff0c;尝试重置Face ID时&#xff0c;系统会弹窗提示“面容ID不可用&#xff0c;稍后尝试设置面容ID。” 如果你的iPhone在没有摔落手机或是手机进水的情况下出现这个弹窗&#xff…

【uniapp小程序】路由跳转navigator传参封装

文章目录&#x1f34d;前言&#x1f34b;正文1、看官网1.1 navigator API 介绍1.2、路由跳转参数传递1.3、五种常见的跳转方式1.3.1 uni.navigateTo(OBJECT)1.3.2 uni.redirectTo(OBJECT)1.3.3 uni.reLaunch(OBJECT)1.3.4 uni.switchTab(OBJECT)1.3.5 uni.navigateBack(OBJECT)…

图的初识·基本概念

文章目录基本概念图有两种基本形式无向图的表示有向图的表示基本概念 图结构也是数据结构的一部分。而且还有一点小难。图是由多个结点链接而成的&#xff0c;但是一个结点可以同时连接多个其他结点&#xff0c;多个节点也可以同时指向一个节点。【多对多的关系】 图结构是任意…

如何从报表控件FastReport .NET中连接到 PostgreSQL 数据库?

FastReport.NET官方版下载 Fastreport是目前世界上主流的图表控件&#xff0c;具有超高性价比&#xff0c;以更具成本优势的价格&#xff0c;便能提供功能齐全的报表解决方案&#xff0c;连续三年蝉联全球文档创建组件和库的“ Top 50 Publishers”奖。 慧都科技是Fast Repor…

mysql索引类别和失效场景

首先&#xff0c;我们为什么要使用索引&#xff0c;索引有什么作用呢&#xff1f; 索引可以用来快速查询数据表中有某一特定值的记录&#xff0c;大大加快数据的查询速度&#xff1b;在列上创建了索引之后&#xff0c;查找数据时可以直接根据该列上的索引找到对应记录行的位置…