【优选算法】位运算 {位运算符及其优先级;位运算的应用:判断位,打开位,关闭位,转置位,位图,get lowbit,close lowbit;相关编程题解析}

一、位运算符及其优先级

我们知道,计算机中的数在内存中都是以二进制形式进行存储的 ,而位运算就是直接对整数在内存中的二进制位进行操作,因此其执行效率非常高,在程序中尽量使用位运算进行操作,这会大大提高程序的性能。

那么,涉及位运算的运算符如下表所示:
在这里插入图片描述

注意:参与位运算的对象只能是整型数据(int, unsigned, char),不能为实型

异或运算符的运算律

  1. 异或0:a^0 = a;
  2. 消消乐:a^a = 0;
  3. 交换律和结合律:abc = acb; abc = a(bc);

位运算符的优先级

优先级需要弄清楚,如果不太清楚可以加小括号确保是想要的运算顺序,这里只是相对优先级,即只是和一些常用的算术运算符做比较。
在这里插入图片描述


二、位运算的应用

2.1 判断位

给一个数n,确定它的二进制表示中的第x位是0还是1

算法:(n>>x)&1

原理:按位与的运算规则——有0则0,遇1不变

举例:判断第3位

  • 11001010 //n
    00011001 //n>>3
    00000001 //&1
    00000001 //结果

2.2 打开位

将一个数n二进制表示的第x位修改成1

算法:n |= (1<<x)

原理:按位或的运算规则——有1则1,遇0不变

举例:打开第4位

  • 11001010 //目标
    00010000 //1<<4
    11011010 //|= 得结果

2.3 关闭位

将一个数n二进制表示的第x位修改成0

算法:n &= ~(1<<x)

原理:按位与的运算规则——有0则0,遇1不变

举例:关闭第7位

  • 11001000 //目标
    10000000 //1<<7
    01111111 //~(1<<7)
    01001000 //&= 得结果

2.4 转置位

将一个数n二进制表示的第x位转置(0变1,1变0)

算法:n ^= (1<<x)

原理:按位异或的运算规则——0变1,1变0

举例:转置第4位

  • 11011010 //目标
    00010000 //1<<4
    11001010 //^= 得结果

2.5 位图

在这里插入图片描述

定义和特性:位图是一种数据结构,它由一个二进制位数组或者比特数组组成,每个位(bit)只能存储 0 或 1。位图通常被用来表示大量整数的集合,通过位的状态来表示该整数是否出现在集合中。位图支持高效的位操作(如按位与、按位或、按位异或等),适合用于高效地存储和操作大量的布尔型数据。

实现方式:位图可以被实现为一个数组,其中每个元素都是一个字节,而每个字节又包含8个位。对于很大的位图,可以使用多个数组进行分段存储。位图通常不仅仅用于表示整数的集合,还可以用于标记某个特定值是否存在。

详细内容请阅读文章:【高阶数据结构】哈希的其他应用 {位图的介绍及实现,位图的组合应用;布隆过滤器的介绍及实现,布隆过滤器的应用;哈希切分}-CSDN博客


2.6 get lowbit

提取一个数n二进制表示中最低位的1

算法:n & -n

原理:-n(按位取反再加1)将最低位的1左边的位(不包括logbit位)全部转置,左边转置后按位与得0,右边相同位按位与不变。

举例:

  • 11011000 //目标
    00101000 //-n
    00001000 //& 得结果

2.7 close lowbit

关闭一个数n二进制表示中最低位的1

算法:n & (n-1)

原理:n-1 将最低位的1右边的位(包括logbit位)全部转置,左边相同位按位与不变,右边转置后按位与得0。

举例:

  • 11011000 //目标
    11010111 //n-1
    11010000 //& 得结果

2.8 其他

  • 位运算实现乘除法:将a左移一位实现*2,将a右移一位实现/2。

    • a < < n ≡ a ∗ 2^n
    • a > > n ≡ a / 2^n
  • 统计二进制数中有多少个1:

    int func(int x){int count=0;while (x){count++;x=x&(x-1); //将lowbit关闭}return count;
    }
    
  • 字符的大小写转换:对应大小写字母ASSIC码的二进制数只有第5位不同(大写为0,小写为1),因此可以通过操作第5位实现大小的相互转换

    • 字符小写转大写:ch&=~32;
    • 字符大写转小写:ch|=32;

    • 字符大小写互换:ch^=32;

  • 交换两变量的值:a=a^b; b=a^b; a=a^b;(异或运算符的运算律:结合律、消消乐)


三、相关编程题

3.1 位1的个数

题目链接

191. 位1的个数 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

不断的close lowbit,每关闭一个1count就++,直到将所有的1都关闭,量的值变为0。

编写代码

class Solution {
public:int hammingWeight(int n) {int count = 0;while(n>0){n &= (n-1);++count;}return count;}
};

3.2 比特位计数

题目链接

338. 比特位计数 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

同上 close lowbit

编写代码

class Solution {
public:vector<int> countBits(int n) {vector<int> ans(n+1, 0);for(int i = 0; i <= n; ++i){int count = 0;int k = i;while(k>0){++count;k &= k-1;}ans[i] = count;}return ans;}
};

3.3 汉明距离

题目链接

461. 汉明距离 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

异或运算:相异为1,相同为0。tmp = x^y将x和y二进制表示中不同的位置置为1,相同的位置置为0。再统计tmp中位1的个数即可,原理同上 close lowbit

编写代码

class Solution {
public:int hammingDistance(int x, int y) {int tmp = x ^ y;int count = 0;while(tmp>0){++count;tmp &= tmp-1;}return count;}
};

3.4 只出现一次的数字

题目链接

136. 只出现一次的数字 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

异或运算的运算律:消消乐

编写代码

class Solution {
public:int singleNumber(vector<int>& nums) {int ret = 0;for(auto e : nums){ret ^= e;}return ret;}
};

3.5 只出现一次的数Ⅲ

题目链接

260. 只出现一次的数字 III - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

  1. 将所有的数按位异或到tmp,最终其实就是只出现一次的两个数的异或结果,即将两个数不同的位置置为1。
  2. get lowbit将tmp的最低位1取出(还是存入tmp)。注意有符号数需要特殊处理INT_MIN(符号位为1,其他位为0),有符号数取反的位操作是:符号位不变,其他位取反再+1。INT_MIN取反会溢出。实际上INT_MIN的lowbit就是它本身。
  3. 再次遍历数组,判断所有数的tmp位(异或结果的最低位1),将只出现一次的两个数分到不同的组中进行异或,最终就能分别得到这两个数了。

编写代码

class Solution {
public:vector<int> singleNumber(vector<int>& nums) {int tmp = 0;for(auto e : nums){tmp ^= e;}tmp = tmp==INT_MIN? tmp:tmp&(-tmp); //get lowbitvector<int> ret(2, 0);for(auto e : nums){if(e & tmp){ret[0] ^= e;}else{ret[1] ^= e;}}return ret;}
};

3.6 判定字符是否唯一

题目链接

面试题 01.01. 判定字符是否唯一 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

在这里插入图片描述

编写代码

//用一个int数据充当位图
class Solution {
public:bool isUnique(string astr) {// 利用鸽巢原理进行优化if(astr.size() > 26)return false;// 用位图判断一个字符是否在集合中int bitmap = 0;for(auto e : astr){int tmp = 1 << (e-'a');if(!(bitmap & tmp)){bitmap |= tmp;}else{return false;}}return true;}
};//用STL的bitset数据结构
class Solution {
public:bool isUnique(string astr) {if(astr.size() > 26)return false;bitset<26> bs;for(auto e : astr){int x = e - 'a'; //位图的第x位if(!bs.test(x)){bs.set(x);}else{return false;}}return true;}
};

3.7 丢失的数字

题目链接

268. 丢失的数字 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

在这里插入图片描述

编写代码

class Solution {
public:int missingNumber(vector<int>& nums) {int ret = 0;int i = 0;for(; i < nums.size(); ++i){ret ^= nums[i];ret ^= i;}ret ^= i;return ret;}
};

3.8 两整数之和

题目链接

371. 两整数之和 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

在这里插入图片描述

编写代码

class Solution {
public:int getSum(int a, int b) {int bitsum = a ^ b; //无进位相加的结果int bitcarry = (a & b)<<1; //进位的结果while(bitcarry != 0) //当不再有进位时退出循环{a = bitsum;b = bitcarry;bitsum = a ^ b;bitcarry = (a & b)<<1;}return bitsum;}   
};

3.9 只出现一次的数字Ⅱ

题目链接

137. 只出现一次的数字 II - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

依次确定每一个二进制位

为了方便叙述,我们称「只出现了一次的元素」为「答案」。

由于数组中的元素都在 int(即 32 位整数)范围内,因此我们可以依次计算答案的每一个二进制位是 0 还是 1。

具体地,考虑答案的第 i 个二进制位(i 从 0 开始编号),它可能为 0 或 1。对于数组中非答案的元素,每一个元素都出现了 3 次,对应着第 i 个二进制位的 3 个 0 或 3 个 1,无论是哪一种情况,它们的和都是 3 的倍数(即和为 0 或 3)。因此:

答案的第 i 个二进制位就是数组中所有元素的第 i 个二进制位之和除以 3 的余数。

这样一来,对于数组中的每一个元素 x,我们使用位运算 (x >> i) & 1得到 x 的第 i 个二进制位,并将它们相加再对 3 取余,得到的结果一定为 0 或 1,即为答案的第 i 个二进制位。

编写代码

class Solution {
public:int singleNumber(vector<int>& nums) {int ret = 0;for(size_t i = 0; i < 32; ++i){int bitsum = 0;for(auto e : nums){bitsum+=(e>>i)&1;}bitsum %= 3;ret |= (bitsum<<i);}return ret;}
};

3.10 消失的两个数字

题目链接

面试题 17.19. 消失的两个数字 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

在这里插入图片描述

编写代码

class Solution {
public:vector<int> missingTwo(vector<int>& nums) {int tmp = 0;//异或到一起for(auto e : nums) tmp ^= e; //原数组for(int i = 1; i<=nums.size()+2; ++i) tmp ^= i; //全数组//此时的tmp种存放的是消失的两个数字的异或结果int lowbit = tmp==INT_MIN? tmp:tmp&(-tmp);vector<int> ret(2, 0);//划分成两类异或for(auto e : nums) //原数组if(e & lowbit) ret[0] ^= e;else ret[1] ^= e;for(int i = 1; i<=nums.size()+2; ++i) //全数组if(i & lowbit) ret[0] ^= i;else ret[1] ^= i; return ret;}
};

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

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

相关文章

04_前端三大件JS

文章目录 JavaScript1.JS的组成部分2.JS引入2.1 直接在head中通过一对script标签定义脚本代码2.2创建JS函数池文件&#xff0c;所有html文件共享调用 3.JS的数据类型和运算符4.分支结构5.循环结构6.JS函数的声明7.JS中自定义对象8.JS_JSON在客户端使用8.1JSON串格式8.2JSON在前…

Python项目开发实战:工厂库存管理系统(案例教程)

一、项目背景与意义 随着制造业的快速发展,工厂库存管理成为了企业运营中不可或缺的一部分。一个高效的库存管理系统能够确保物料供应的及时性、降低库存成本、提高生产效率。因此,我们决定使用Python开发一个工厂库存管理系统,以满足工厂日常库存管理的需求。 二、系统需求…

Flutter 中的 Badge 小部件:全面指南

Flutter 中的 Badge 小部件&#xff1a;全面指南 在移动应用设计中&#xff0c;徽章&#xff08;Badge&#xff09;是一种常见的UI元素&#xff0c;用于吸引用户注意并展示重要信息&#xff0c;如未读消息数量、新通知等。Flutter 通过各种第三方包提供了徽章小部件&#xff0…

弘君资本股市行情:股指预计保持震荡上扬格局 关注汽车、银行等板块

弘君资本指出&#xff0c;近期商场体现全体分化&#xff0c;指数层面上看&#xff0c;沪指一路震动上行&#xff0c;创出年内新高&#xff0c;创业板指和科创50指数体现相对较弱&#xff0c;依然是底部震动走势。从盘面体现上看&#xff0c;轮动依然是当时商场的主基调&#xf…

IBERT眼图扫描(高速收发器八)

前文讲解了GTX的时钟及收发数据通道的组成&#xff0c;之后讲解了眼图、加重、均衡等原理及原因&#xff0c;本文通过xilinx提供的IBERT IP完成实际工程的眼图扫描&#xff0c;确定加重和幅值调节的参数。 1、回环模式 在此之前&#xff0c;需要了解一下GTX的回环模式。如果板…

【字典树(前缀树) 字符串】2416. 字符串的前缀分数和

本文涉及知识点 字典树&#xff08;前缀树) 字符串 LeetCode 2416. 字符串的前缀分数和 给你一个长度为 n 的数组 words &#xff0c;该数组由 非空 字符串组成。 定义字符串 word 的 分数 等于以 word 作为 前缀 的 words[i] 的数目。 例如&#xff0c;如果 words [“a”,…

【list】list库介绍 + 简化模拟实现

本节博客先对list进行用法介绍&#xff0c;再在库的基础上简化其内容和形式&#xff0c;简单进行模拟实现&#xff0c;有需要借鉴即可。 目录 1.list介绍1.1 list概述1.2相关接口的介绍 2.简化模拟实现3.各部分的细节详述3.1结点3.2迭代器细节1&#xff1a;迭代器用原生指针还是…

深入理解TCP滑动窗口协议

在计算机网络中&#xff0c;传输控制协议&#xff08;TCP&#xff09;是确保可靠数据传输的核心协议之一。TCP的可靠性和高效性部分得益于其滑动窗口机制。本文将详细介绍TCP滑动窗口协议的工作原理及其在实际中的应用。 什么是滑动窗口协议&#xff1f; 滑动窗口协议是一种用…

【Codesys】-执行第三方程序,或Windows脚本

该记录旨在解决RTE作为第一个Windows的一个exe程序不能调用其他程序的问题。 可以实现:在PLC界面打开第三方程序、在PLC界面关闭本机Windows操作系统 首先添加依赖库-SysProcess,3.5.17.0 然后在程序里执行相应的指令&#xff0c;该指令可以被Windows识别为类似于执行Bat文件…

国产操作系统上使用rsync恢复用户数据 _ 统信 _ 麒麟 _ 中科方德

原文链接&#xff1a;国产操作系统上使用rsync恢复用户数据 | 统信 | 麒麟 | 中科方德 Hello&#xff0c;大家好啊&#xff01;今天给大家带来一篇关于在国产操作系统上使用rsync备份并还原用户数据的文章。rsync是一款功能强大的文件同步和备份工具&#xff0c;广泛用于Linux系…

河南乙级道路与桥梁资质年审材料准备要点解析

河南乙级道路与桥梁资质年审材料准备要点解析如下&#xff1a;河南宽信权经理 一、企业基本情况材料 营业执照副本复印件&#xff1a;确保复印件清晰、完整&#xff0c;并加盖企业公章。 企业章程&#xff1a;提供最新的企业章程&#xff0c;并加盖企业公章。此材料需反映企业…

代码随想录——从前序与中序遍历序列构造二叉树(Leetcode105)

题目链接 递归 /*** Definition for a binary tree node.* public class TreeNode {* int val;* TreeNode left;* TreeNode right;* TreeNode() {}* TreeNode(int val) { this.val val; }* TreeNode(int val, TreeNode left, TreeNode right) {* …

Linux(三)

Linux&#xff08;三&#xff09; Linux网络配置管理网络基础知识 IP地址A类 由1个字节网络地址3个字节主机地址B类 由2个字节网络地址2个主机地址C类 由3个字节网络地址1个主机地址D类:主要用于组播E类:为将来使用保留 子网掩码子网掩码作用网关DNS服务器 Linux用户管理用户的…

如何看centos 有没有安装x11

在CentOS系统中&#xff0c;可以通过检查是否存在X11相关的包来判断是否安装了X11。你可以使用yum工具来查询是否安装了xorg-x11-server-Xorg包&#xff0c;这通常是X11服务器的包名。 打开终端&#xff0c;输入以下命令&#xff1a; yum list installed | grep xorg-x11-ser…

浅析部署架构中的GZone、RZone和CZone

在现代软件开发中&#xff0c;理解和应用各种技术概念是成功的重要因素。本文将详细介绍GZone、RZone和CZone三个概念&#xff0c;解释它们的定义、特点、功能及应用场景&#xff0c;并通过实际案例帮助读者更好地理解这些概念。 一、GZone 1.1 定义 GZone是指“Global Zone…

MTK Android9.0 给vendor下文件夹权限,用于读取文件列表

1.背景 最近在TV开发中遇到一个问题&#xff1a;如何判断设备烧录过HDCP KEY的问题&#xff0c;由于MTK的官方接口返回值并不准确&#xff0c;只能判断2.2是否烧录&#xff0c;不能准确判断1.4是否烧录过&#xff0c;因为HDCP 的KEY有两个&#xff0c;分别是1.4和2.2&#xff…

结构体常见问题

问题一: 结构体的自引用 错误: typedef struct _Pos{ int x; int y; struct _Pos z; }pos; 定义一个结构体&#xff0c;其内部又定义了一个同类的结构体变量&#xff0c;这是一种错误。(因为如果你在结构体内部定义自己类型的结构体变量&#xff0c;在定义结构体的…

Shell脚本零碎知识积累

1、使用While循环打印目录中的所有文件&#xff1a; #!/bin/bash# 定义要遍历的目录 directory"./"# 使用while循环读取目录中的文件 while read file doecho "Found file: $file" done < <(find "$directory" -maxdepth 1 -type f)最后一…

linux创建离线yum源给局域网机器使用

适用场景&#xff1a;在封闭的内网环境中&#xff0c;无法使用互联网进行安装各种rpm包的时候&#xff0c;离线yum源可以解决大部分问题&#xff0c;配置号后可直接使用yum进行安装包 1.准备好镜像源ISO&#xff1a; 例如以下示例&#xff0c;具体可参考自己的系统进行下载&a…

新书推荐:7.3 for语句

本节必须掌握的知识点&#xff1a; 示例二十四 代码分析 汇编解析 7.3.1 示例二十四 ■for语句语法形式&#xff1a; for(表达式1;表达式2;表达式3) { 语句块; } ●语法解析&#xff1a; 第一步&#xff1a;执行表达式1&#xff0c;表达式1初始化循环变量&#xff1b; …