位运算(位运算的技巧、二进制中1的个数、区间或、异或森林)

 一、移位操作符

1.1   左移操作符  <<

作用:二进制数向左边移动,右边补0.

#include<stdio.h>
int main()
{
    int a = 10;
    int b = a << 1;//将a的二进制向左移动一位
    printf("a=%d\nb=%d", a, b);
    return 0;
}

左移操作符相当于对原数进行乘以2的幂次方的操作

对于整数5(二进制表示为00000101),执行左移三位操作,相当于执行 5 * (2^{3})。

1.2   右移操作符>>

右移操作符分为逻辑右移和算数右移

逻辑右移:左边用零填充,右边丢弃
算术右移:左边用原该值的符号位填充,右边丢弃

由于大部分编译器及代码默认为算数右移。

右移相当于对原数进行除以2的幂次方的操作

例如,对于整数13(二进制表示为00001101),执行左移2位操作,相当于执行13/4向下取整。

1.3、位操作符

&    按位与:只要有0就是0,两个同时为1才是1。

|     按位或:只要有1就是1,两个同时为0才是0。

^    按位异或:相同为0,相异为1。

~    按位取反:将一个数的二进制位0取1,1取0,之后再加一。

异或的性质:

交换律:x^y=y^x

结合律:x^(y^ z)= (x^y)^z

自反性:x^x=0

零元素:x^0=x

逆运算:x^y=z,则有z^y=x(两边同时异或y,抵消掉)

//二进制计算时用补码计算
int main()
{
	int a = 3;
	int b = -5;
	int c = a & b;
	/*按(二进制)位与运算
	计算规则:对应二进制位进行与运算
	只要有0就是0,两个同时为1才是1
	00000000000000000000000000000011 --- 3的补码
	11111111111111111111111111111011 --- -5的补码
	00000000000000000000000000000011 --- 结果
	*/
	printf("c = %d\n", c);
}

int main()
{
	int a = 3;
	int b = -5;

	int d = a | b;
	/*按(二进制)位或运算
	计算规则:对应二进制位进行或运算
	只要有1就是1,两个同时为0才是0
	00000000000000000000000000000011 --- 3的补码
	11111111111111111111111111111011 --- -5的补码
	11111111111111111111111111111011 --- 结果
	*/
	printf("d = %d\n", d);


	return 0;
}

int main()
{
	int a = 3;
	int b = -5;
	int e = a ^ b;
	/*按(二进制)位异或运算
	计算规则:对应二进制位进行异或运算
	相同为0,相异为1
	00000000000000000000000000000011 --- 3的补码
	11111111111111111111111111111011 --- -5的补码
	11111111111111111111111111111000 --- 结果
	10000000000000000000000000000111 --- 反码
	10000000000000000000000000001000 --- 补码
	*/
	printf("e = %d\n", e);


	return 0;
}


int main()
{
	int a = 3;
	int b = -5;
	
	int f = ~a;
	/*求补码
	00000000000000000000000000000011 -- 3的原码
	11111111111111111111111111111100(补码)
	00000000000000000000000000000011
	00000000000000000000000000000100 > -4
	*/
	printf("e = %d", f);

	return 0;
}

二、位运算的技巧

1.1编写代码实现:求一个整数存储在(**内存中**)的二进制中1的个数

普通写法:

int main()
{
	int a = -1; int i = 0,count = 0;
	//a & 1 == 1;就说明a的二进制中最低位是1
	//a & 1 == 0;就说明a的二进制中最低位是0
	//a >> 1;依次顺序移动遍历二进制中的每一位
	for (i = 0; i < 32; i++)
	{
		(a >> i) & 1;
		count++;
	}
	printf("%d", count);
	return 0;
}

考虑正负

int count_one_bit(unsigned int n)//把有符号当成无符号数
{
	int count = 0;
	
	while (n)
	{
		if (n % 2 == 1)
			count++;
		n = n / 2;
	}
	return count;
}

int main()
{
	int num = 0,t = 0;
	scanf("%d", &num);
	//求一个整数存储在内存中的二进制中1的个数
	t = count_one_bit(num);
	printf("%d", t);
	return 0;
}

1.2  n & (n - 1)的运用,求一个整数存储在内存中的二进制中1的个数

/*n & (n - 1)的运用*/
int count_one_bit(unsigned int n)//把有符号当成无符号数
{
	int count = 0;

	while (n)
	{
		n = n & (n - 1);
		//效果:把二进制中最右边的1去掉了
		//n = 15
		//1111 - n	  1110 - n-1
		//1110 - n    1101 - n-1
		//1100 - n    1011 - n-1
		//1000 - n    0111 - n-1
		//0000 - n
		count++;
	}
	return count;
}

int main()
{
	int num = 0, t = 0;
	scanf("%d", &num);
	//求一个整数存储在内存中的二进制中1的个数
	t = count_one_bit(num);
	printf("%d", t);
	return 0;
}

1.3判断奇偶

   x & 1
// 如果结果为 1 说明是奇数
// 结果为 0 说明是偶数

1.4 获取二进制数的某一位

   x >> i & 1;
// 结果必然为0或1, 表示 x 的二进制表示中的第i位

1.5修改二进制中的某一位

   x | (1 << i)
// 将 x 的第i位或上1, 则x[i]变为1,
// 其他位上或上0没有影响

1.6 快速判断一个数字是否为2的幂次方

   x & (x - 1)
// 如果 x 为2的幂次方, 则 x 的二进制表示中只有一个1
// x - 1 就有很多个连续的1并且和 x 的1没有交集,
// 两者与运算一定为0, 可以证明其他情况必然不为0

1.7 获取二进制中最低位的1

   lowbit(x) = x & (-x)
// 如果 x = (010010), 则lowbit(x) = (000010)
// 常用于数据结构树状数组

三、二进制中1的个数

题目描述
给定一个整数x,输出该数二进制表示中1的个数。
例: 9的二进制表示为 1001,有2位是1,所以函数返回 2。
输入描述
输入 x  (内存空间为 32 位的整数)
输出描述
第一行输出 x 二进制表示中1的个数。
输入输出样例

#include <iostream>
using namespace std;
int main()
{
  unsigned int x; cin >> x;
  int ans = 0;
  while(x)
  {
    if(x & 1)ans++;
    x >>= 1;
  }
  cout << ans << '\n';
  return 0;
}

四、区间或

问题描述

给定一个长度为 n 的数组a。现在有 q 次询问,给出两个整数 l 和 r ,求a[l];|;a[ l+ 1];|......|;a[r-1];|;a[r]的值,其中|代表按位或。

输入格式

第一行包含两个整数 n 和 q,分别表示数组的长度和询问的次数。
第二行包含 n 个整数,a1,a2...an表示数组 a 的元素。
接下来的 q 行,每行包含两个整数 l 和 r,表示一个询问的区间。
输出格式

对于每次询问,输出一个整数表示结果。


#include <bits/stdc++.h>
using namespace std;

const int N = 1e5 + 9;
int a[N], prefix[35][N];

int main()
{
	int n, q; cin >> n >> q;
	for (int i = 1; i <= n; ++i)cin >> a[i];

	for (int w = 0; w <= 30; ++w)
		for (int i = 1; i <= n; ++i)
		{
			prefix[w][i] = prefix[w][i - 1] + ((a[i] >> w) & 1);
		}
	while (q--)
	{
		int l, r; cin >> l >> r;
		int ans = 0;
		for (int w = 0; w <= 30; ++w)
		{
			ans += (1 << w) * (prefix[w][r] - prefix[w][l - 1] ? 1 : 0);
		}
		cout << ans << '\n';
	}
	return 0;
}

五、异或森林

问题描述
在一个神秘的世界中,存在着一个称为"异或森林”的地方。异或森林中的每个树木都拥有独特的力量。肖恩进入了这片森林,他得到了一个任务:找出数组中满足条件的连续子数组,使得连续子数组中所有元素异或运算结果的因数个数为偶数。完成任务将揭示宝藏的所在地。现在,你能告诉肖恩有多少个连续子数组满足条件吗?
注意:0 的因数个数视为奇数。

输入描述
第一行输入一个数字 n 表示数组元素个数。
第二行输入 n 个数字,第 i 个数字 a[i] 表示数组的第i个元素
数据保证 1 ≤ n ≤ 1e4,1 ≤ a[i]≤n。

#include <bits/stdc++.h>
using namespace std;

const int N = 1e5 + 9;
int a[N], prexor[N], cnt[N];

int main()
{
	int n; cin >> n;
	for (int i = 1; i <= n; ++i)cin >> a[i];
	for (int i = 1; i <= n; ++i)prexor[i] = prexor[i - 1] ^ a[i];
	cnt[0] = 1; int ans = n * (n + 1) / 2;
	for (int i = 1; i <= n; ++i)
	{
		for (int j = 0; j <= 200; ++j)
		{
			int sq = j * j;
			ans -= cnt[prexor[i] ^ sq];
		}
		cnt[prexor[i]]++;
	}
	cout << ans << '\n';
	return 0;
}

今天就先到这了!!!

看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注!

你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。

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

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

相关文章

语文教学方法有哪些,产生了什么效果

你是否曾想过&#xff0c;一位普通的语文老师如何化身为智慧的引导者&#xff0c;点燃学生心中的求知之火&#xff1f;让我们一起探寻那些神奇的语文教学方法&#xff0c;以及它们带来的深远影响。 不仅让知识变得容易理解&#xff0c;更在无形中培养了学生的各项能力。通过谈话…

华中某科技大学校园网疑似dns劫持的解决方法

问题 在校园网ping xxx.ddns.net&#xff0c;域名解析失败 使用热点ping xxx.ddns.net&#xff0c;可以ping通 尝试设置windows dns首选dns为114.114.114.114&#xff0c;重新ping&#xff0c;仍然域名解析失败 猜测【校园网可能劫持dns请求】 解决方法 使用加密的dns请求…

制作照片数字人,让图片开口说话

一、软件安装 全部按照默认程序安装 最后安装完成后把所有按钮取消勾选&#xff0c;选择【Finish】&#xff0c;在弹出的界面中选择【否】。 然后点击管理员运行&#xff0c;直接全部默认&#xff0c;不更改位置。 二、制作数字人 根据自己的电脑位数&#xff0c;选择打开 选…

修改一个教材上的网站源码使它能在www服务器子目录上正常运行

修改一个教材上的网站源码&#xff0c;使它能在www服务器子目录上正常运行。 该网站源码是教材《PHPMySQL网站开发项目式教程》上带的网站源码。该源码是用 php html 写的。该源码包含对mysql数据库进行操作的php代码。以前该网站源码只能在www服务器的根目录上正常运行&…

【Docker】技术架构演变

【Docker】技术架构演变 目录 【Docker】技术架构演变架构中的概念架构演进单机架构相关软件 应用数据分离架构应用服务集群架构相关软件 读写分离/主从分离架构相关软件 引入缓存——冷热分离架构相关软件 垂直分库&#xff08;分布式数据库架构&#xff09;相关软件 业务拆分…

第八篇 - 预测受众(Predictive audience)技术是如何赋能数字化营销生态的?- 我为什么要翻译介绍美国人工智能科技巨头IAB公司

IAB平台&#xff0c;使命和功能 IAB成立于1996年&#xff0c;总部位于纽约市。 作为美国的人工智能科技巨头社会媒体和营销专业平台公司&#xff0c;互动广告局&#xff08;IAB- the Interactive Advertising Bureau&#xff09;自1996年成立以来&#xff0c;先后为700多家媒…

Android制作.9图回忆

背景 多年前&#xff0c;做app开发遇到IM需求&#xff0c;那会用到.9图做聊天气泡背景&#xff0c;现在总结下使用png图片制作.9图。方法有很多&#xff0c;这里主要介绍Android studio制作.9图。当然使用ps、draw9patch都行。 第一步、打开Android studio&#xff0c;切换到dr…

【Linux实践室】Linux常用命令:文件操作|文件夹操作

&#x1f308;个人主页&#xff1a;聆风吟 &#x1f525;系列专栏&#xff1a;Linux实践室、网络奇遇记 &#x1f516;少年有梦不应止于心动&#xff0c;更要付诸行动。 文章目录 一. ⛳️任务描述二. ⛳️相关知识2.1 &#x1f514;Linux文件操作2.1.1 &#x1f47b;创建文件2…

【粉丝福利】一本书讲透ChatGPT,实现从理论到实践的跨越!大模型技术工程师必读

&#x1f33c;一、前言 OpenAI 在 2022 年 11 月推出了人工智能聊天应用—ChatGPT。它具有广泛的应用场景&#xff0c;在多项专业和学术基准测试中表现出的智力水平&#xff0c;不仅接近甚至有时超越了人类的平均水平。这使得 ChatGPT 在推出之初就受到广大用户的欢迎&#xf…

速卖通关键字搜索API接口实战:Python代码与搜索策略解析

一、速卖通关键字搜索API简介 速卖通&#xff08;AliExpress&#xff09;作为阿里巴巴旗下的国际电商平台&#xff0c;为卖家和买家提供了便捷的交易渠道。其开放平台提供的API接口允许开发者集成速卖通的各种功能&#xff0c;其中之一就是关键字搜索API。通过这个API&#xf…

QT计算两个日期之间的月份数

数据库中单表数据存储量过大时&#xff0c;会造成数据库的查询统计速度变慢&#xff0c;因此需将单表数据拆分存储到按年月命名的多张数据表中。解决思路是获取单表中的最小时间和最大时间&#xff0c;然后计算两个时间中的月份数量&#xff0c;最后根据开始年月循环算出所有需…

蓝牙系列四:开源蓝牙协议BTStack框架分析

在学习蓝牙的时候,最好还是要借助源代码来学习,并且进行实际的操作从而更加理解蓝牙协议栈的运行。依然根据韦东山老师的学习视频进行学习,整理记录如下。 首先,协议栈是跑在硬件上的,我们这里选择使用windows来操控硬件。来看一下硬件的结构: 蓝牙模块接在电脑上,或是…

(每日持续更新)jdk api之PipedWriter基础、应用、实战

博主18年的互联网软件开发经验&#xff0c;从一名程序员小白逐步成为了一名架构师&#xff0c;我想通过平台将经验分享给大家&#xff0c;因此博主每天会在各个大牛网站点赞量超高的博客等寻找该技术栈的资料结合自己的经验&#xff0c;晚上进行用心精简、整理、总结、定稿&…

设置word目录从正文开始记录页码,并解决word目录正常,但正文页脚处只显示第一页的页码

设置word目录从正文开始记录页码&#xff0c;并解决word目录正常&#xff0c;但正文页脚处只显示第一页的页码 问题详情1&#xff1a;如何设置目录从正文开始记录页码 问题详情2&#xff1a;word目录处的页码正常&#xff0c;但正文只有第一页的页脚处显示页码 解决方法 在设置…

web开发:如何用Echarts来自动给网页设计各种统计图

很多时候web开发也会需要用到统计图&#xff0c;如果单纯靠我们自己那点拙劣的css和js水平设计的话&#xff0c;又耗时间又做得跟史一样&#xff0c;这时候就需要引入别人设计师为我们设计好的动态统计图——echarts Echarts的官网是&#xff1a;Apache ECharts 1、第一步&…

稀碎从零算法笔记Day4-LeetCode:交替合并字符串

前言&#xff1a;今天妹有深夜档&#xff0c;因为8点有个飞机 题型&#xff1a;字符串、双指针&#xff08;笔者没用这个思路&#xff09; 链接&#xff1a;1768. 交替合并字符串 - 力扣&#xff08;LeetCode&#xff09; 来源&#xff1a;LeetCode 著作权归作者所有。商业转…

时钟域交叉设计——Clock Domain Crossing Design

What is Metastability? 任何关于时钟域交叉&#xff08;CDC&#xff09;的讨论&#xff0c;都应从对可变性和同步性的基本了解开始。通俗地说&#xff0c;可变性是指一种不稳定的中间状态&#xff0c;在这种状态下&#xff0c;最轻微的干扰也会导致稳定状态的恢复。当应用于…

QT----QTcreater连接Mysql数据库

目录 1、下载驱动&#xff0c;放入文件夹2、编写代码&#xff0c;实现本地访问3、实现网络数据库3.1 更改权限3.2 修改代码 之前写了一个图书管理系统用的是sqlite3&#xff0c;现在想用mysql&#xff0c;部署到网上&#xff0c;实现远程访问。 1、下载驱动&#xff0c;放入文…

Draft-P802.11be-D3.2协议学习__$Annex-Z-HE-SIG-B-and-EHT-SIG-content-examples

Draft-P802.11be-D3.2协议学习__$Annex-Z-HE-SIG-B-and-EHT-SIG-content-examples Z.1 GeneralZ.2 HE-SIG-B example 1Z.3 HE-SIG-B example 2Z.4 HE-SIG-B example 3Z.5 HE-SIG-B example 4Z.6 EHT-SIG example 1&#xff08;EHT OFDMA 80MHz&#xff09;Z.7 EHT-SIG example …

虚假交易商常态化,2月下半月FX110曝光43家黑平台

以半个月为期&#xff0c;FX110网对虚假交易商进行常态化曝光&#xff0c;只为极力打压黑平台生存空间&#xff0c;让更多人的避免被骗。 2月上半月&#xff0c;FX110网曝光黑平台41家&#xff0c;下半月曝光数与上半月基本持平&#xff0c;为43家。 曝光的这43家黑平台绝大部…