题目:给定范围 [m, n], 0 <= m <= n <= 2147483647,返回此范围内所有数字的按位与(包含 m, n 两端点)。
输入: [5,7] 输出: 4
输入: [0,1] 输出: 0
找这些数字的公共前缀,进一步可以转换成最小和最大两个数的公共前缀。
为什么只用 选取最大和最小两个数 ?
因为最大和最小两个数之间,其他数字,会将公共前缀后的所有位,0与1都会取到
而可能第二个最小的数进位后,和最大的数的公共前缀,比最小数和最大数的公共前缀不一致(反之,亦当如此)
因此,选取最大和最小两个数,能保证其最终查询到的公共前缀,就是区间内所有数字的公共前缀
public int rangeBitwiseAnd(int m, int n) {
int shiftBits = 0;
while (m < n) { // 不是从m遍历到n,而是移位,最多32次
m = m >> 1;
n = n >> 1;
shiftBits++;
}
return m << shiftBits;
}
时间复杂度: O(1)。虽然算法中有一个循环,但是迭代次数是由整数的位数限定的,所以迭代次数是固定的。因此,算法的时间复杂度是常数级别的。
空间复杂度: O(1),不管输入是什么,内存消耗是常数的。