六十、走进位运算的大门

共 2712字,需浏览 6分钟

 ·

2020-12-20 07:23


「@Author:Runsen」

编程的本质来源于算法,而算法的本质来源于数学,编程只不过将数学题进行代码化。「---- Runsen」

根据我的脑海大纲,现在应该进入位运算的大门。

位运算,计算机内所有的数都以二进制存储,位运算是对二进制位的操作。

按位“与”运算

按位“与”运算的运算符是:&。首先我声明一个变量i等于11,j等于2,然后我们计算i和j的按位“与”运算,也就是11&2,得到的结果是2.‘

个2是怎么来的,首先我们要把这个11和这个2全部转换成二进制,我们从上图看出11的二进制是0b1011,2的二进制是0b100b这个它就是一个二进制的标识位,就是告诉我们这个是二进制。

按照位“与”运算的原则,如果两个值相应的位置都为1也就是上下都为1的时候,那么该结果就是1,如果有一个不是1,那么就是0

1011
0010
----
0010

所以,1011和0010的按位“与”运算就是0010,然后将二进制的0010转化为十进制,就是我们的结果2。

按位“或”运算

下面,我们介绍按位“或”运算。按位“或”运算的运算符是:|。我们同样计算11和2的按位“或”运算,得到的结果是11。

这个结果2到底是怎么样来进行运算的?11的二进制是1011,而2的二进制是0010。

按照位“或”运算的原则,如果两个值相应的位置有一个是1,那么该结果就是1,也就是如果都是0,该结果就是0

1011
0010
----
1011

所以,1011和0010的按位“与”运算就是1011,然后将二进制的1011转化为十进制,就是我们的结果11。

按位“异或”运算

下面,我们介绍按位“异或”运算。按位“异或”运算的运算符是:^。我们同样计算11和2的 按位“异或”运算,得到的结果是9。

这个结果9到底是怎么样来进行运算的?11的二进制是1011,而2的二进制是0010。

按位“异或”运算的原则,如果两个值相应的位置相同1,那么该结果就是0,也就是如果都是0或者都是1,该结果就是0.同样只有是1和0,那些结果就是1

1011
0010
----
1001

所以,1011和0010的按位“异或”运算就是1001,然后将二进制的1001转化为十进制,就是我们的结果9。

按位取反运算

下面,我们介绍按位取反运算。按位“异或”运算的运算符是:~。我们计算11的按位取反运算,得到的结果是-12。

这个结果-12到底是怎么样来进行运算的?按位取反的计算结论是:~n = -(n+1)

因此,11的按位取反运算 :~11=-(11+1)=-12

左移运算符

下面,我们介绍左移运算符。左移运算符是:<<。我们计算11的左移2位的运算,得到的结果是44。

这个结果44到底是怎么样来进行运算的?左移2位,就是1011往左移动2位,然后用0来补充,变成了101100。本来在这里向左边移动,其实是后边舍弃掉2位

1011
----
101

所以,1011的右移2位的运算就是101100,然后将二进制的101100转化为十进制,就是我们的结果44。

右移运算符

下面,我们介绍右移运算符。右移运算符是:>>。我们计算11的右移2位的运算,得到的结果是2。

这个结果2到底是怎么样来进行运算的?右移2位,就是1011往右移动2位,然后右边的11直接去除,变成了10。本来在这里向右边移动,其实是后边删除位数。

1011
----
10

所以,1011和的左移2位的运算就是10,然后将二进制的10转化为十进制,就是我们的结果2。下表就是总结。

符号意义示例
~逻辑非运算~a
&逻辑与运算a&b
|逻辑或运算a|b
<<位左移运算a<<2
>>位右移运算a>>2

判断奇数还是偶数

通常判断奇数还是偶数我们想到的办法就是除以2,看余数是否为0。具体Python代码如下:

def isodd(x):
    return True if (x % 2 == 0else False

那么如何使用位运算呢?

我们只需要使用&运算,与1进行&,如果为1,那么该数为奇数;如果为0,那么该数是偶数,具体Python代码如下:

def isodd(x):
    return True if (x & 1else False

左移一位相当于乘以2,右移一位相当于除以2

在面试的过程中,通常会遇到的一个问题是写二分查找代码。

Python二分查找的代码如下:

def binary_search(list, item):
    '''
    :param list: 有序列表
    :param item: 要查找的元素
    :return: item在list中的索引,若不在list中返回None
    '''

    low = 0
    high = len(list) - 1
    while low <= high:
        mid = (low + high) // 2
        if list[mid] == item:
            return mid
        elif list[mid] < item:
            low = mid + 1
        elif list[mid] > item:
            high = mid - 1
    return None

其中有一步是需要取最小下标和最大下标的中间值,若使用位运算符,则变成midpoint = (low + high) >> 1,面试官肯定会对你刮目相看。

下面就是位运算常见用法总结

「判断x的奇偶数」

x % 2 == 1 // 常规写法
x & 1 == 1 // 为真奇数,为假偶数

「计算中位数」

mid = (left + right) / 2  // 常规写法
mid  = (left + right) >> 1 // 位运算

LeetCode 第 78题:子集

下面,我们来看一题关于二进制的Leetcode。题目很简单,就是给定一组不含重复元素的整数数组 nums,返回该数组所有可能的子集(幂集)

#给定一组不含重复元素的整数数组 nums,返回该数组所有可能的子集(幂集)。 
# 说明:解集不能包含重复的子集。 
# 示例: 
# 输入: nums = [1,2,3]
#输出:
#[
#  [3],
#  [1],
#  [2],
#  [1,2,3],
#  [1,3],
#  [2,3],
#  [1,2],
#  []
#] 
# Related Topics 位运算 数组 回溯算法

既然这道题让我们求的是所有的子集,那么我们可以从子集的特点入手。假如整数数组的长度是3,那么子集有8个。整数数组的长度是4,那么子集有16个。

一个含有n个元素的子集的数量是,这个是高中数学中的集合的数学题,不知道的可能忘得一干二净。

这题非常巧妙地使用二进制的位运算。使用位运算,对每一位为1的表示这一位的数字存在,求[1,2,3] 的子集,我们可以用0和1表示有和无。比如编码001,表示只含有3,编码101,表示含有1,3。编码111表示1,2,3

因为每个数只有选与不选两种情况,具体的情况如下所示。

具体的Python代码。

class Solution:
    def subsets(self, nums: List[int]) -> List[List[int]]:
        ret = []
        n = len(nums)
        # 遍历所有的状态
        # 二进制的巧用:1左移n位相当于2的n次方,也可以写成2**n
        for s in range(1 << n):
            cur = []
            # 通过位运算找到每一位是0还是1
            for i in range(n):
                # 判断s状态在2的i次方上,也就是第i位上是0还是1
                if s & (1 << i):
                    cur.append(nums[i])
            ret.append(cur[:])
        return ret

其实位运算说简单,也很简单,但有的时候要用,我却怎么也想不到,主要是因为奇葩很多。

在此题中,还有一种方法是回溯算法,注意需要将空集的情况添加。

class Solution:
    def subsets(self, nums: List[int]) -> List[List[int]]:
        res = [[]]
        for i in nums:
            res = res + [[i] + num for num in res]
        return res

本文已收录 GitHub,传送门~[1] ,里面更有大厂面试完整考点,欢迎 Star。



Reference

[1]

传送门~: https://github.com/MaoliRUNsen/runsenlearnpy100


更多的文章

点击下面小程序



- END -


浏览 23
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报