02 位运算

本文介绍了Python中的位运算,包括原码、反码和补码概念,按位非、与、或、异或操作,以及左移和右移的技巧。通过实例演示了如何利用位运算快速计算和整数集合操作,并以LeetCode题目为例,展示了如何运用位运算解决实际问题。特别提到了Python中负数表示的特殊性,以及如何正确处理补码输出。

转载自Datawhale开源学习库

https://github.com/datawhalechina/team-learning-program/tree/master/PythonLanguage

位运算

1. 原码、反码和补码

二进制有三种不同的表示形式:原码、反码和补码,计算机内部使用补码来表示

原码:就是其二进制表示(注意,最高位是符号位)。

00 00 00 11 -> 3
10 00 00 11 -> -3

反码:正数的反码就是原码,负数的反码是符号位不变,其余位取反(对应正数按位取反)。

00 00 00 11 -> 3
11 11 11 00 -> -3

补码:正数的补码就是原码,负数的补码是反码+1。

00 00 00 11 -> 3
11 11 11 01 -> -3

符号位:最高位为符号位,0表示正数,1表示负数。在位运算中符号位也参与运算。

2. 按位非操作 ~

~ 1 = 0
~ 0 = 1

~num的补码中的 0 和 1 全部取反(0 变为 1,1 变为 0)有符号整数的符号位在 ~ 运算中同样会取反。

00 00 01 01 -> 5
~
---
11 11 10 10 -> -6

11 11 10 11 -> -5
~
---
00 00 01 00 -> 4

3. 按位与操作 &

1 & 1 = 1
1 & 0 = 0
0 & 1 = 0
0 & 0 = 0

只有两个对应位都为 1 时才为 1

00 00 01 01 -> 5
&
00 00 01 10 -> 6
---
00 00 01 00 -> 4

4. 按位或操作 |

1 | 1 = 1
1 | 0 = 1
0 | 1 = 1
0 | 0 = 0

只要两个对应位中有一个 1 时就为 1

00 00 01 01 -> 5
|
00 00 01 10 -> 6
---
00 00 01 11 -> 7

5. 按位异或操作 ^

1 ^ 1 = 0
1 ^ 0 = 1
0 ^ 1 = 1
0 ^ 0 = 0

只有两个对应位不同时才为 1

00 00 01 01 -> 5
^
00 00 01 10 -> 6
---
00 00 00 11 -> 3

异或操作的性质:满足交换律和结合律

A: 00 00 11 00
B: 00 00 01 11

A^B: 00 00 10 11
B^A: 00 00 10 11

A^A: 00 00 00 00
A^0: 00 00 11 00

A^B^A: = A^A^B = B = 00 00 01 11

6. 按位左移操作 <<

num << inum的二进制表示向左移动i位所得的值。

00 00 10 11 -> 11
11 << 3
---
01 01 10 00 -> 88 

7. 按位右移操作 >>

num >> inum的二进制表示向右移动i位所得的值。

00 00 10 11 -> 11
11 >> 2
---
00 00 00 10 -> 2 

8. 利用位运算实现快速计算

通过 <<>> 快速计算2的倍数问题。

n << 1 -> 计算 n*2
n >> 1 -> 计算 n/2,负奇数的运算不可用
n << m -> 计算 n*(2^m),即乘以 2 的 m 次方
n >> m -> 计算 n/(2^m),即除以 2 的 m 次方
1 << n -> 2^n

通过 ^ 快速交换两个整数。

a ^= b
b ^= a
a ^= b

通过 a & (-a) 快速获取a的最后为 1 位置的整数。

00 00 01 01 -> 5
&
11 11 10 11 -> -5
---
00 00 00 01 -> 1

00 00 11 10 -> 14
&
11 11 00 10 -> -14
---
00 00 00 10 -> 2

9. 利用位运算实现整数集合

一个数的二进制表示可以看作是一个集合(0 表示不在集合中,1 表示在集合中)。

比如集合 {1, 3, 4, 8},可以表示成 01 00 01 10 10 而对应的位运算也就可以看作是对集合进行的操作。

元素与集合的操作:

a | (1<<i)  -> 把 i 插入到集合中
a & ~(1<<i) -> 把 i 从集合中删除
a & (1<<i)  -> 判断 i 是否属于该集合(零不属于,非零属于)

集合之间的操作:

a 补   -> ~a
a 交 b -> a & b
a 并 b -> a | b
a 差 b -> a & (~b)

注意:整数在内存中是以补码的形式存在的,输出自然也是按照补码输出。

【例子】C#语言输出负数。

class Program
{
    static void Main(string[] args)
    {
        string s1 = Convert.ToString(-3, 2);
        Console.WriteLine(s1); 
        // 11111111111111111111111111111101
        
        string s2 = Convert.ToString(-3, 16);
        Console.WriteLine(s2); 
        // fffffffd
    }
}

【例子】 Python 的bin() 输出。

print(bin(3))  # 0b11
print(bin(-3))  # -0b11

print(bin(-3 & 0xffffffff))  
# 0b11111111111111111111111111111101

print(bin(0xfffffffd))       
# 0b11111111111111111111111111111101

print(0xfffffffd)  # 4294967293

是不是很颠覆认知,我们从结果可以看出:

  • Python中bin一个负数(十进制表示),输出的是它的原码的二进制表示加上个负号,巨坑。
  • Python中的整型是补码形式存储的。
  • Python中整型是不限制长度的不会超范围溢出。

所以为了获得负数(十进制表示)的补码,需要手动将其和十六进制数0xffffffff进行按位与操作,再交给bin()进行输出,得到的才是负数的补码表示。


练习题

leetcode 习题 136. 只出现一次的数字

给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素。

尝试使用位运算解决此题。

题目说明:

"""
Input file
example1: [2,2,1]
example2: [4,1,2,1,2]

Output file
result1: 1
result2: 4
"""



class Solution:
    def singleNumber(self, nums: List[int]) -> int:
        
     # your code here
&lt;think&gt;我们参考站内引用,结合位运算的基础知识和应用场景来回答用户的问题。 用户询问位运算的应用场景和实际运用案例。从引用中我们可以总结出以下内容: 1. 引用[1]介绍了位运算的基本规则:与(&amp;)、或(|)、异或(^)、取反(~)、左移(&lt;&lt;)、右移(&gt;&gt;)等运算规则。 2. 引用[2]给出了一个指定位数比较的例子:通过位运算提取整数中的特定位。 3. 引用[3]详细描述了位运算在颜色处理中的应用:提取RGB颜色分量以及合并RGB颜色值。 4. 引用[4]提到了位运算在数据结构(异或双向链表)中的应用,以及一些位运算表达式的计算。 因此,我们可以围绕这些方面展开回答,并适当补充其他常见应用场景。 应用场景: 1. 位操作在底层编程、嵌入式系统、图形处理、密码学等领域应用广泛。 2. 位运算可以提高程序的执行效率,因为位运算直接对内存中的二进制位进行操作,通常比四则运算快。 具体案例: 案例1:颜色处理(如引用[3]) - 提取RGB分量:通过右移和与运算提取颜色中的红、绿、蓝分量。 - 合并RGB分量:通过左移和或运算将三个分量合并成一个整数值。 案例2:权限系统(经典应用) - 使用一个整数的不同二进制位表示不同的权限(如读、写、执行等)。 - 例如:定义权限:读(1&lt;&lt;0)、写(1&lt;&lt;1)、执行(1&lt;&lt;2)。 - 给用户分配权限:使用或运算组合权限,如读和写:permission = READ | WRITE。 - 检查权限:使用与运算,如 if (permission &amp; READ) ... 表示有读权限。 案例3:数据压缩与存储 - 使用位来存储布尔值,可以大大节省空间。例如,一个32位整数可以存储32个布尔值。 - 设置第n位:value |= (1 &lt;&lt; n) - 清除第n位:value &amp;= ~(1 &lt;&lt; n) - 切换第n位:value ^= (1 &lt;&lt; n) 案例4:交换两个变量的值(不使用临时变量) - 使用异或运算:a ^= b; b ^= a; a ^= b; 案例5:判断奇偶性 - 使用与运算:if (n &amp; 1) 为奇数,否则为偶数。 案例6:乘除2的幂 - 左移一位相当于乘以2:n &lt;&lt; 1 - 右移一位相当于除以2:n &gt;&gt; 1 (注意有符号数的符号位扩展问题) 案例7:异或双向链表(引用[4]) - 为了节省空间,可以用一个指针域存储前后两个指针的异或值,从而实现双向链表。 案例8:位掩码(引用[2]) - 通过掩码来检查、设置、清除特定的位。 案例9:求绝对值(针对32位整数) - 利用位运算:int mask = (n &gt;&gt; 31); 然后 (n + mask) ^ mask; 或者 (n ^ mask) - mask; 案例10位运算在加密算法中的应用 - 例如在AES等加密算法中,使用位运算进行置换和混淆。 注意:位运算在处理有符号数时需要注意符号位的影响,特别是右移操作(有符号右移&gt;&gt;和无符号右移&gt;&gt;&gt;)。 下面我们给出一些简单的代码示例(以Java/C++/Python等类似语法)来演示部分案例: 示例1:权限系统 ```java // 定义权限 final int READ = 1 &lt;&lt; 0; // 0001 final int WRITE = 1 &lt;&lt; 1; // 0010 final int EXECUTE = 1 &lt;&lt; 2; // 0100 // 设置权限 int userPermission = READ | WRITE; // 0011 // 检查写权限 if ((userPermission &amp; WRITE) != 0) { System.out.println(&quot;有写权限&quot;); } // 添加执行权限 userPermission |= EXECUTE; // 去除写权限 userPermission &amp;= ~WRITE; ``` 示例2:交换两个数 ```python a = 5 b = 3 a = a ^ b b = a ^ b # b = (a ^ b) ^ b = a a = a ^ b # a = (a ^ b) ^ a = b print(a, b) # 输出3 5 ``` 示例3:提取RGB分量(引用[3]) ```java // 24位颜色,0xff00cc int color = 0xff00cc; int r = (color &gt;&gt; 16) &amp; 0xFF; // 0xff int g = (color &gt;&gt; 8) &amp; 0xFF; // 0x00 int b = color &amp; 0xFF; // 0xcc ``` 示例4:合并RGB分量 ```java int r = 0xff; int g = 0x00; int b = 0xcc; int color = (r &lt;&lt; 16) | (g &lt;&lt; 8) | b; // 0xff00cc ``` 示例5:判断奇偶 ```c int n = 5; if (n &amp; 1) { printf(&quot;奇数&quot;); } else { printf(&quot;偶数&quot;); } ``` 通过这些案例,我们可以看到位运算的灵活性和高效性。在实际编程中,合理运用位运算可以提升程序性能并减少内存占用。 注意:虽然位运算高效,但会降低代码可读性,因此建议在必要的地方使用并加上清晰的注释。 &sect;&sect;相关问题&sect;&sect; 1. 位运算在算法竞赛中有什么常用的技巧? 2. 如何用位运算实现两个整数的加法? 3. 位运算在加密算法中的具体应用有哪些? &lt;/think&gt; ### 位运算的应用场景与实际运用案例 位运算直接操作二进制位,在底层系统、算法优化和特定场景中具有高效、节省资源的优势。以下是典型应用场景和案例: --- #### 一、基础位运算规则回顾 | 运算符 | 描述 | 运算规则(二进制) | |--------|--------------|----------------------------------------| | `&amp;` | 与 | 两位均为1时结果为1:$0\&amp;0=0,\ 1\&amp;1=1$ | | `|` | 或 | 两位有1时结果为1:$0\|0=0,\ 1\|1=1$ | | `^` | 异或 | 两位不同时结果为1:$0^1=1,\ 1^1=0$ | | `~` | 取反 | 按位取反:$~0=1,\ ~1=0$ | | `&lt;&lt;` | 左移 | 高位丢弃,低位补0 | | `&gt;&gt;` | 右移 | 低位丢弃,高位补符号位 | | `&gt;&gt;&gt;` | 无符号右移 | 低位丢弃,高位补0 | --- #### 二、实际应用场景及案例 1. **颜色处理(RGB/A 色彩编码)** - **提取颜色分量**(引用[^3]): ```java // 提取24位RGB颜色分量 int color24 = 0xff00cc; int r = (color24 &gt;&gt; 16) &amp; 0xFF; // 右移16位取红色分量 (0xff) int g = (color24 &gt;&gt; 8) &amp; 0xFF; // 右移8位取绿色分量 (0x00) int b = color24 &amp; 0xFF; // 直接取蓝色分量 (0xcc) // 提取32位ARGB颜色分量(含透明度) int color32 = 0xff342388; int a = color32 &gt;&gt;&gt; 24; // 无符号右移取透明度 (0xff) ``` - **合并颜色值**: ```java int r = 0xff, g = 0x00, b = 0xcc; int rgb = (r &lt;&lt; 16) | (g &lt;&lt; 8) | b; // 合并为0xff00cc ``` 2. **权限控制系统** 用二进制位表示不同权限(如读、写、执行): ```python READ = 0b001 # 1 WRITE = 0b010 # 2 EXEC = 0b100 # 4 user_perm = READ | EXEC # 赋予读和执行权限 (0b101) # 检查是否具有写权限 if user_perm &amp; WRITE: print(&quot;有写权限&quot;) else: print(&quot;无写权限&quot;) # 输出&quot;无写权限&quot; ``` 3. **高效数学运算** - **判断奇偶性**:`n &amp; 1 == 0` 为偶数(等价于 `n % 2 == 0`) - **乘除2的幂**: - $n \times 2^k \rightarrow n &lt;&lt; k$(如 `5 &lt;&lt; 2 = 20`) - $n / 2^k \rightarrow n &gt;&gt; k$(如 `20 &gt;&gt; 2 = 5`) - **交换变量值**(无临时变量): ```c a ^= b; // a = a ^ b b ^= a; // b = b ^ (a ^ b) = a a ^= b; // a = (a ^ b) ^ a = b ``` 4. **数据结构优化** - **异或双向链表**(引用[^4]): 用单指针存储相邻节点的异或值,节省内存: ```c struct Node { int data; Node* np; // np = prev ^ next }; // 遍历(从头到尾) Node *prev = NULL, *curr = head; while (curr != tail) { Node *next = (Node*)((uintptr_t)prev ^ (uintptr_t)curr-&gt;np); prev = curr; curr = next; } ``` 5. **位掩码(Bitmask)** - **提取特定位**(引用[^2]): ```java int n = 8; // 二进制: 00001000 int bit = (n &amp; 0b1000) &gt;&gt; 3; // 提取第4位 &rarr; 1 ``` - **标志位管理**: ```c #define FLAG_A 0x01 // 00000001 #define FLAG_B 0x02 // 00000010 int flags = 0; flags |= FLAG_A; // 设置A标志 flags &amp;= ~FLAG_B; // 清除B标志 ``` --- #### 三、典型问题解析 1. **表达式求值**(引用[^4]): - `(char)(127 &lt;&lt; 1) + 1` $127_{(10)} = 01111111_{(2)}$,左移1位得 $11111110_{(2)} = -2$(有符号char),加1得 $-1$。 - `1 &lt;&lt; 2 + 3` 先计算加法:$2+3=5$,再左移:$1 &lt;&lt; 5 = 32$(优先级:`+`高于`&lt;&lt;`)。 --- #### 四、优势与注意事项 - **优势**:执行速度快、内存占用低(如用1位存储布尔值)。 - **注意事项**: - 右移运算符`&gt;&gt;`对有符号数补符号位,`&gt;&gt;&gt;`补0(无符号)。 - 避免未定义行为(如负数的左移)。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值