当前位置: 首页 > news >正文

【C语言刷题】——初识位操作符

【C语言刷题】——初识位操作符

    • 位操作符介绍
    • 题一、 不创建临时变量(第三个变量),实现两个数的交换
      • (1)法一
      • (2)法二
    • 题二、 求一个数存储在内存中的二进制中“一”的个数
      • (1)法一
      • (2)法二
      • (3)法三
    • 题三、 单身狗1
      • (1)法一
      • (2)法二
    • 题四、 单身狗二
      • (1)法一
      • (2)法二
      • (3)法三

位操作符介绍

位操作符有:

<<      //左移操作符
>>      //右移操作符
&       //按位与
|       //按位或
^       //按位异或
~       //按位取反

  注:他们的操作数必须是整数。

  更多关于位操作符介绍请看这篇文章:【C语言】——详解操作符(上)

题一、 不创建临时变量(第三个变量),实现两个数的交换

(1)法一

  
参考代码:

#include<stdio.h>int main()
{int a = 0;int b = 10;a = a + b;b = a - b;a = a - b;printf("交换后a = %d b = %d\n", a, b);return 0;
}

  
代码讲解:

  相信这段代码大家都能看懂,这里我就不多解释了。
  
  但遗憾的是这个方法有一点问题:当 a 和 b 的值很大(但都不超过 i n t int int 的存储空间)时,他们相加会超过 int 的存储空间,导致丢失数据。
  
  那有什么更好的办法吗?有的,请看法二。

  

(2)法二

  
参考代码:

#include<stdio.h>int main()
{int a = 0;int b = 10;a = a ^ b;b = a ^ b;a = a ^ b;printf("交换后a = %d b = %d", a, b);return 0;
}

  
代码讲解:

首先,我们需掌握几个知识点:

  • 一个数与他自身异或为 0 0 0 ,即: a a a ^ a a a == 0 0 0
  • 任何数与 0 0 0 异或结果都为其本身,即: a a a ^ 0 0 0 == 0 0 0
  • 异或运算满足交换律,即: a a a ^ a a a ^ b b b == a a a ^ b b b ^ a a a

  第八行代码:b = a ^ b;,由于第七行代码:a = a ^ b; ,此时a = a ^ b;,即b = a ^ b ^ b,得b = a ^ 0 ;得 b = a

  同理,第九行代码:a = a ^ b;,由于第七行代码:a = a ^ b; 和第八行代码结果b = a,即:a = a ^ b ^ a,得 a = b

  我们可以这样来理解:我们把第七行代码a = a ^ b;当成是一把钥匙,碰到 a 得 b碰到 b 得 a

  

题二、 求一个数存储在内存中的二进制中“一”的个数

  

(1)法一

  
参考代码:

#include <stdio.h>int main()
{int num = 0;scanf("%d", &num);int count = 0;//计数while (num){if (num % 2 == 1)count++;num = num / 2;}printf("二进制中1的个数 = %d\n", count);return 0;
}

  
代码讲解:

  其实,该法的解题思路与在十进制中打印每一位是思路相同。在十进制中,要想获得每一位,我们的方法是:先余十,再除十,不断循环。这里,也是一样的只是因为是二进制改为先余二,再除而,不断循环,遇到结果为 1 ,则计数器加一。
  
  但是这个方法有点问题:它只能处理正数的情况,如果是负数,他就没办法了。为什么?因为:if(num % 2 == 1),负数余二永远不可能为 1,但num = num / 2;语句正常执行,输入复数的结果永远是 0。

  

(2)法二

  
参考代码:

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

  
代码讲解:

  法二的思路是:给这个数的每一位都与上 1 (不断右移),因为与的逻辑是:有 0 为 0,全 1 为 1。当运算结果为 0 ,表示该位为 0, 结果为 1 ,该位为 1
  
图解(以5为例):
在这里插入图片描述
结果为1,计数器加一
  
5右移一位:
在这里插入图片描述
结果为0,计数器不变
  
5再右移
在这里插入图片描述
结果为1,计数器加一
  
接着不断右移,一共32次,因为后面的结果都是0,变不再一一赘述。
  
  但该方法一定要循环32次,有没有效率更高的方法呢?

  

(3)法三

  
参考代码:

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

  
代码讲解:

该方法的核心:这个数本身与他自身减一与(&) 运算,不断循环
  

n = n & (n - 1);

或许大家一脸疑惑,别急,直接上图!

以15为例:
循环一次
在这里插入图片描述

循环两次:
在这里插入图片描述

循环三次:
在这里插入图片描述

循环四次:
在这里插入图片描述

  大家发现没有,每循环一次就会把最右边的 1 给消去,当最终把所有 1 消去变成零,循环结束,而我们只需要计算循环了多少次就能知道该数有几个 1 。

  

题三、 单身狗1

  
题目:

  在一个整型数组中,只有一个数字出现一次,其他数组都是成对出现的,请找出那个只出现一次的数字。
  
例如:

  数组中有:1 2 3 4 5 1 2 3 4,只有5出现一次,其他数字都出现2次,找出5

  

(1)法一

  
参考代码:

#include<stdio.h>int main()
{int arr[] = { 1, 2, 3, 4, 5, 1, 2, 3, 4 };int sz = sizeof(arr) / sizeof(arr[0]);for (int i = 0; i < sz; i++){int flag = 0;int n = 0;for (int j = 0; j < sz; j++){if (i == j)continue;n = arr[i] ^ arr[j];if (n == 0){flag = 1;break;}}if (flag == 0)printf("单身狗是:%d\n", arr[i]);}return 0;
}

  
代码讲解:

  该方法想必大家都很容易想到,逻辑也很简单:遍历数组中的每一个数,每个数再遍历一遍除自身外的整个数组,遇到与自身一样的数就说明自己不是单身狗。
  这里就不再过多解释

  

(2)法二

  
参考代码:

int main()
{int arr[] = { 1, 2, 3, 4, 5, 1, 2, 3, 4 };int sz = sizeof(arr) / sizeof(arr[0]);int a = 0;for (int i = 0; i < sz ; i++){a = arr[i] ^ a;}printf("单身狗是:%d\n", a);return 0;
}

  
解题思路:

这里再带大家重温一下按位异或的相关知识:
(1)一个数与他自身异或为 0 0 0 ,即: a a a ^ a a a == 0 0 0
(2)任何数与 0 0 0 异或结果都为其本身,即: a a a ^ 0 0 0 == 0 0 0

那么综合运用起来就是这题的解法啦

将数组元素全部异或
  
a = 1 ^ 2 ^ 3 ^ 4 ^ 5 ^ 1 ^ 2 ^ 3 ^ 4
a = 1 ^ 1 ^ 2 ^ 2 ^ 3 ^ 3 ^ 4 ^ 4 ^ 5
a = 0 ^ 0 ^ 0 ^ 0 ^ 0 ^ 0 ^ 0 ^ 0 ^ 5
a = 5

题四、 单身狗二

  
题目:

一个数组中只有两个数字是出现一次,其他所有数字都出现了两次。

编写一个函数找出这两个只出现一次的数字。

  
例如:

有数组的元素是:1,2,3,4,5,1,2,3,4,6

只有5和6只出现1次,要找出5和6.

  

(1)法一

参考代码:

#include<stdio.h>int main()
{int arr[] = { 1, 2, 3, 4, 5, 1, 2, 3, 4,6 };int sz = sizeof(arr) / sizeof(arr[0]);int count = 0;for (int i = 0; i < sz; i++){int flag = 0;int n = 0;for (int j = 0; j < sz; j++){if (i == j)continue;n = arr[i] ^ arr[j];if (n == 0){flag = 1;}}if (0 == flag){printf("单身狗是:%d\n", arr[i]);count++;}if (2 == count)break;}return 0;
}

  
代码讲解:

  这段代码的思路与上一题单身狗一的法一思路是相同的,
  
  即依次取出数组中的每个元素,让他与数组中剩下的元素比较,当遍历完整个数组依然没找到相同的数时,说明他是其中一个单身狗,计数器 +1 ,并打印;当计数器为二时,说明已找到全部单身狗,退出循环。

  

(2)法二

参考代码:

#include<stdio.h>int main()
{int arr[] = { 1,2,3,4,5,1,2,3,4,6 };int sz = sizeof(arr) / sizeof(arr[0]);int i = 0;for (i = 0; i < 32; i++){int j = 0;int last_0 = 0;int last_1 = 0;for (j = 0; j < sz; j++){if ((arr[j] & (1 << i)) == 0){last_0 ^= arr[j];}else if ((arr[j] & (1 << i)) != 1){last_1 ^= arr[j];}}if(last_0 != 0 && last_1 != 0){printf("%d %d", last_0, last_1);break;}}return 0;
}

  
代码讲解:

  做了上面的单身狗,我们想这题应该也可以用异或分方法来解,
  
  我们想到,如果将所有元素直接异或,肯定是无法直接找出两只单身狗,这时我们可以想到先将他们分组,让每一组都只有一只单身狗,再将两组全部异或就行了。那么怎么分组呢?
  
  我们看到因为数组两两元素相同,加上两只单身狗,所以数组总数一定是偶数,这时我们可以依次遍历数组中所有元素的二进制位数,将该位为 1 和为 0 的元素分成两组,分完组后,若两组的元素个数都为奇数(相同的数该位一定相同,一定被分到同一组,剩下一只单身狗,为奇数),则成功将两只单身狗分开,再分别异或两组中的所有元素就能找出两周单身狗啦,
  
  如果分完组两边都是偶数,则比较二进制下一位,直到分出奇数组。

  

(3)法三

参考代码:

#include<stdio.h>void findTwoNum(int arr[], int n, int * pnum1, int * pnum2)
{int i;int sum = 0;for (i = 0; i < 9; i++){sum ^= arr[i];} //先找到两个数互相异或的结果int pos;for (i = 0; i < 32; i++){if (sum & 1 << i){pos = i;break;}} //再找到有分歧的一位。在这一位上,两个数一定是一个1一个0*pnum1 = *pnum2 = 0;for (i = 0; i < 10; i++){if (arr[i] & 1 << pos){*pnum1 ^= arr[i]; //这一位是1的,放在数1里}else{*pnum2 ^= arr[i]; //这一位是0的,放在数2里}}
}

  
代码讲解:

  法二虽然用到了异或操作,但分起组来太过麻烦,效率并不高,有没有什么方法能实现快速分组呢?答案当然是有的。

  我们知道:两个相同的数异或为 0,将数组中的所有元素异或起来,得到的结果就是两只单身狗异或的结果。
  
  
  
  这时我们可能就要问了,那得到这个结果有什么用呢?肯定不止唯二这两个数异或才得出这个结果,其他两个数异或也有可能得到这个结果。

  确实如此,但我们别忘了异或的特点:相同为 0,相异为 1。我们只需要找到异或的结果的其中一个为“ 1 ”的位数,这说明在这个位,其中一只单身狗是0,另一只为1。

  这时,我们只需要将数组中的元素分两组:一组的元素在该位的值为 0,另一组该位值为 1,再将两组的所有元素分别异或起来,自然就得到两只单身狗啦。怎么样,是不是很巧妙呢。

相关文章:

【C语言刷题】——初识位操作符

【C语言刷题】——初识位操作符 位操作符介绍题一、 不创建临时变量&#xff08;第三个变量&#xff09;&#xff0c;实现两个数的交换&#xff08;1&#xff09;法一&#xff08;2&#xff09;法二 题二、 求一个数存储在内存中的二进制中“一”的个数&#xff08;1&#xff0…...

Python 对Excel工作表中的数据进行排序

在Excel中&#xff0c;排序是整理数据的一种重要方式&#xff0c;它可以让你更好地理解数据&#xff0c;并为进一步的分析和报告做好准备。本文将介绍如何使用第三方库Spire.XLS for Python通过Python来对Excel中的数据进行排序。包含以下三种排序方法示例&#xff1a; 按数值…...

Python对头发二维建模(考虑风力、重力)

目录 一、背景 二、代码 一、背景 数值方法被用于创建电影、游戏或其他媒体中的计算机图形。例如&#xff0c;生成“逼真”的烟雾、水或爆炸等动画。本文内容是对头发的模拟&#xff0c;要求考虑重力、风力的影响。 假设&#xff1a; 1、人的头部是一个半径为10厘米的球体。…...

Python基础快速入门

Python基础快速入门 前置知识 Python Python是一种广泛使用的高级编程语言&#xff0c;以其易于学习和使用的语法而闻名。以下是Python的一些主要特点&#xff1a; 高级语言&#xff1a;Python是一种高级语言&#xff0c;这意味着它提供了较高层次的抽象&#xff0c;使编程更…...

C++的学习

代码练习 输入一个字符串&#xff0c;统计其中大写字母、小写字母、数字、空格以及其他字符的个数 #include <iostream>using namespace std;int main() {cout << "请输入一个字符串" << endl;string str;getline(cin,str);int capital 0;int l…...

工地安全反光衣穿戴监测报警摄像机

工地安全反光衣穿戴监测报警摄像机是为了提高工地施工人员的安全意识和监管效率而设计的。这种设备结合了反光衣、监测系统和报警摄像机的功能&#xff0c;可以有效减少工地事故的发生。 首先&#xff0c;工地安全反光衣是一种具有高度可见度的服装&#xff0c;能够使穿戴者在夜…...

UNIAPP微信小程序中使用Base64编解码原理分析和算法实现

为何要加上UNIAPP及微信小程序&#xff0c;可能是想让检索的翻围更广把。&#x1f607; Base64的JS原生编解码在uni的JS引擎中并不能直接使用&#xff0c;因此需要手写一个原生的Base64编解码器。正好项目中遇到此问题&#xff0c;需要通过URLLink进行小程序跳转并携带Base64参…...

人工智能|机器学习——K-means系列聚类算法k-means/ k-modes/ k-prototypes/ ......(划分聚类)

1.k-means聚类 1.1.算法简介 K-Means算法又称K均值算法&#xff0c;属于聚类&#xff08;clustering&#xff09;算法的一种&#xff0c;是应用最广泛的聚类算法之一。所谓聚类&#xff0c;即根据相似性原则&#xff0c;将具有较高相似度的数据对象划分至同一类簇&#xff0c;…...

注意力、自注意力和多头注意力的区别

本文作者&#xff1a; slience_me 注意力、自注意力和多头注意力的区别 理解注意力&#xff08;Attention&#xff09;、自注意力&#xff08;Self-Attention&#xff09;和多头注意力&#xff08;Multi-Head Attention&#xff09;之间的区别非常重要&#xff0c;因为它们是自…...

FTP,SFTP,FTPS,SSL,TSL简介,区别,联系,使用场景说明

文章目录 简介FTPFTPSSFTP加密场景选择FTPS还是SFTPFTP、SFTP、FTPS区别、联系和具体使用场景如何使用FTP、SFTP和FTPSSSLTLSSSL和TLS区别和联系&#xff0c;以及使用场景SSL和TLS技术上的区别一些问题隐式的TLS&#xff08;FTPS/SSL&#xff09;或者显式的TLS&#xff08;FTPS…...

路由算法与路由协议

路由选择协议的核心是路由算法&#xff0c;即需要何种算法来获得路由表中的各个项目。 路由算法的目的很简单&#xff1a;给定一组路由器以及连接路由器的链路&#xff0c;路由算法要找到一条从源路由器到目标路由器的最佳路径。通常&#xff0c;最佳路径是指具有最低费用的路…...

dubbo接口自动化用例性能优化

前言 去年换了一个新部门&#xff0c;看了下当前的自动化用例的情况&#xff0c;发现存在三类性能问题&#xff1a; 本地调试运行时等待时间较长&#xff0c;就算是一个简单的case&#xff0c;执行时间都需要1分钟以上单用例执行时间比较长&#xff0c;部分用例执行时间超过2…...

.net core框架

ASP.NET Core 入门 跨平台开源框架 B/S 类与方法 Console 部分称为“类”。 类“拥有”方法&#xff1b;或者可以说方法存在于类中。 WriteLine() 部分称为“方法”。 想要使用方法就要知道方法在哪里 —————————— 执行流 一次执行一段 ASP.NET Core 是什么东西…...

学习大数据,所需要Java基础(9)

文章目录 网络编程实现简答客户端和服务器端的交互编写客户端编写服务端 文件上传文件上传客户端以及服务器端实现文件上传服务器端实现&#xff08;多线程&#xff09;文件上传服务器端&#xff08;连接池版本&#xff09;关闭资源工具类 BS架构服务器案例案例分析BS结构服务器…...

Python元组(Tuple)深度解析!

目录 1. 什么是元组&#xff1f; 2. 创建元组 3.访问元组 4.元组的运算 5.修改元组不可行 6.元组的应用场景 前面的博客里&#xff0c;我们详细介绍了列表&#xff08;List&#xff09;这一种数据类型&#xff0c;现在我们来讲讲与列表相似的一种数据类型&#xff0c;元组…...

排序 Comparable接口、Comparator接口

String类的Comparable接口 1、String类实现了Comparable<String>接口&#xff0c;并提供了compareTo方法的实现&#xff0c;因此&#xff0c;字符串对象&#xff08;即String类型的实例&#xff09;可以直接调用compareTo()方法来比较它们。2、String类的compareTo()方法…...

得帆助力大族激光主数据平台建设,用数据为企业生产力赋能

本期客户 大族激光科技产业集团股份有限公司&#xff08;以下简称“大族激光”&#xff09;是一家从事工业激光加工设备与自动化等配套设备及其关键器件的研发、生产、销售&#xff0c;激光、机器人及自动化技术在智能制造领域的系统解决方案的优质提供商&#xff0c;是国内激光…...

实名认证电子签署:防范合同纠纷,提升交易信任

当今社会&#xff0c;随着数字化和信息化的发展&#xff0c;电子合同已经成为商务活动中常见的签署方式。而在签署电子合同时进行实名认证&#xff0c;是为了确保合同的真实性、合法性和安全性。本文将从法律、技术和实际应用等方面详细解释为什么签署电子合同需要进行实名认证…...

c++ primer中文版第五版作业第十八章

仓库地址 文章目录 18.118.218.318.418.518.618.718.818.918.1018.1118.1218.1318.1418.1518.16位置一using声明 位置二using声明 位置一using指示 位置二using指示 18.1718.1818.1918.2018.2118.2218.2318.2418.2518.2618.2618.2818.2918.30 18.1 此时r是一个range_error类型…...

vue触发真实的点击 事件 跟用户行为一致

<template><div><button ref"myButton" click"handleClick">按钮</button></div> </template><script> export default {methods: {handleClick() {const button this.$refs.myButton;// 创建一个鼠标点击事件…...

内存分配函数malloc kmalloc vmalloc

内存分配函数malloc kmalloc vmalloc malloc实现步骤: 1)请求大小调整:首先,malloc 需要调整用户请求的大小,以适应内部数据结构(例如,可能需要存储额外的元数据)。通常,这包括对齐调整,确保分配的内存地址满足特定硬件要求(如对齐到8字节或16字节边界)。 2)空闲…...

【位运算】消失的两个数字(hard)

消失的两个数字&#xff08;hard&#xff09; 题⽬描述&#xff1a;解法&#xff08;位运算&#xff09;&#xff1a;Java 算法代码&#xff1a;更简便代码 题⽬链接&#xff1a;⾯试题 17.19. 消失的两个数字 题⽬描述&#xff1a; 给定⼀个数组&#xff0c;包含从 1 到 N 所有…...

测试markdown--肇兴

day1&#xff1a; 1、去程&#xff1a;7:04 --11:32高铁 高铁右转上售票大厅2楼&#xff0c;穿过候车厅下一楼&#xff0c;上大巴车 &#xffe5;10/人 **2、到达&#xff1a;**12点多到达寨子&#xff0c;买门票&#xff0c;美团/抖音&#xff1a;&#xffe5;78人 3、中饭&a…...

【快手拥抱开源】通过快手团队开源的 KwaiCoder-AutoThink-preview 解锁大语言模型的潜力

引言&#xff1a; 在人工智能快速发展的浪潮中&#xff0c;快手Kwaipilot团队推出的 KwaiCoder-AutoThink-preview 具有里程碑意义——这是首个公开的AutoThink大语言模型&#xff08;LLM&#xff09;。该模型代表着该领域的重大突破&#xff0c;通过独特方式融合思考与非思考…...

高危文件识别的常用算法:原理、应用与企业场景

高危文件识别的常用算法&#xff1a;原理、应用与企业场景 高危文件识别旨在检测可能导致安全威胁的文件&#xff0c;如包含恶意代码、敏感数据或欺诈内容的文档&#xff0c;在企业协同办公环境中&#xff08;如Teams、Google Workspace&#xff09;尤为重要。结合大模型技术&…...

JDK 17 新特性

#JDK 17 新特性 /**************** 文本块 *****************/ python/scala中早就支持&#xff0c;不稀奇 String json “”" { “name”: “Java”, “version”: 17 } “”"; /**************** Switch 语句 -> 表达式 *****************/ 挺好的&#xff…...

初学 pytest 记录

安装 pip install pytest用例可以是函数也可以是类中的方法 def test_func():print()class TestAdd: # def __init__(self): 在 pytest 中不可以使用__init__方法 # self.cc 12345 pytest.mark.api def test_str(self):res add(1, 2)assert res 12def test_int(self):r…...

【Go语言基础【12】】指针:声明、取地址、解引用

文章目录 零、概述&#xff1a;指针 vs. 引用&#xff08;类比其他语言&#xff09;一、指针基础概念二、指针声明与初始化三、指针操作符1. &&#xff1a;取地址&#xff08;拿到内存地址&#xff09;2. *&#xff1a;解引用&#xff08;拿到值&#xff09; 四、空指针&am…...

深度学习水论文:mamba+图像增强

&#x1f9c0;当前视觉领域对高效长序列建模需求激增&#xff0c;对Mamba图像增强这方向的研究自然也逐渐火热。原因在于其高效长程建模&#xff0c;以及动态计算优势&#xff0c;在图像质量提升和细节恢复方面有难以替代的作用。 &#x1f9c0;因此短时间内&#xff0c;就有不…...

【LeetCode】算法详解#6 ---除自身以外数组的乘积

1.题目介绍 给定一个整数数组 nums&#xff0c;返回 数组 answer &#xff0c;其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法&#xff0c;且在 O…...