给你两个整数 left
和 right
,表示区间 [left, right]
,返回此区间内所有数字 按位与 的结果(包含 left
、right
端点)。
示例 1:
输入:left = 5, right = 7
输出:4
示例 2:
输入:left = 0, right = 0
输出:0
示例 3:
输入:left = 1, right = 2147483647
输出:0
概述 最直观的解决方案就是迭代范围内的每个数字,依次执行按位与运算,得到最终的结果,但此方法在 [m,n][m,n][m,n] 范围较大的测试用例中会因超出时间限制而无法通过,因此我们需要另寻他路。
我们观察按位与运算的性质。对于一系列的位,例如[1, 1, 0, 1, 1],只要有一个零值的位,那么这一系列位的按位与运算结果都将为零。
回到本题,首先我们可以对范围内的每个数字用二进制的字符串表示,例如
,然后我们将每个二进制字符串的位置对齐。
在上图的例子中,我们可以发现,对所有数字执行按位与运算的结果是所有对应二进制字符串的公共前缀再用零补上后面的剩余位。
那么这个规律是否正确呢?我们可以进行简单的证明。假设对于所有这些二进制串,前 iii 位均相同,第
位开始不同,由于
连续,所以第
位在
的数字范围从小到大列举出来一定是前面全部是
,后面全部是
,在上图中对应
均为
,
均为
。并且一定存在连续的两个数
和
,满足
的第
位为
,后面全为
,
的第
位为
,后面全为
,对应上图中的例子即为
和
。这种形如 0111…和 1000…的二进制串的按位与的结果一定为 0000…,因此第
位开始的剩余位均为
,前
位由于均相同,因此按位与结果不变。最后的答案即为二进制字符串的公共前缀再用零补上后面的剩余位。
进一步来说,所有这些二进制字符串的公共前缀也即指定范围的起始和结束数字
和
的公共前缀(即在上面的示例中分别为 999 和 121212)。
因此,最终我们可以将问题重新表述为:给定两个整数,我们要找到它们对应的二进制字符串的公共前缀。
方法一:位移 思路
鉴于上述问题的陈述,我们的目的是求出两个给定数字的二进制字符串的公共前缀,这里给出的第一个方法是采用位移操作。
我们的想法是将两个数字不断向右移动,直到数字相等,即数字被缩减为它们的公共前缀。然后,通过将公共前缀向左移动,将零添加到公共前缀的右边以获得最终结果。
算法
如上述所说,算法由两个步骤组成:
我们通过右移,将两个数字压缩为它们的公共前缀。在迭代过程中,我们计算执行的右移操作数。将得到的公共前缀左移相同的操作数得到结果。
class Solution {
public:
int rangeBitwiseAnd(int m, int n) {
int shift = 0;
// 找到公共前缀
while (m < n) {
m >>= 1;
n >>= 1;
++shift;
}
return m << shift;
}
};
复杂度分析
时间复杂度:
。算法的时间复杂度取决于
和
的二进制位数,由于
,因此时间复杂度取决于
的二进制位数。 空间复杂度:
。我们只需要常数空间存放若干变量。