LeetCode题目:191. 位1的个数
三种方法:
- 常规的方法,每次遍历二进制数中的一位,如果为1则count += 1,这里每次遍历的时候都会采用左移或者右移,但是要注意一个坑, 如果是一个有符号的数,每次右移之后都会不上符号位,因此对于一个有符号的负数,将会一直补上1,只到最后数字变为0xffffffff, 只会陷入死循环, 但是题目中说了是无符号整数,因此我们可以右移。Tips:如果面试官问右移或者左移运算改成除法或者乘法,是不可行的,因为计算机对于位运算的效率远远高于除法或乘法,所以我们平常用乘或除以特定数字的时候可以采用位运算来提升速度,时间复杂度:int对应计算机的位数
- 面试官最喜欢的方法:x = x & (x - 1) 例如对于二进制 10100 将会直接跳过后两个0到最后一个1所在的位置 时间复杂度:二进制表示中1的个数 思路:把一个整数减去1之后再和原来的整数做位与运算,得到的结果相当于把整数的二进制表示中最右边的1变成0
- 自己总结:该方法针对于Python的,先将一个数字转换为二进制,之后转换为字符串,然后遍历字符串中的每一位,统计1的个数,发现竟然比方法2效率还高,
Python内置函数还是强啊
时间复杂度:int对应计算机的位数
面试官最喜欢的方法
# 方法一:对于该题右移不会出现bug,因此我们可以右移
class Solution(object):
def hammingWeight(self, n):
count = 0
while n != 0b00:
# 如果最后一位为1, 否则是不会count+1
if n & 1:
count += 1
n = n >> 1
return count
这里使用c语言实现方法一的左移版本,不使用python原因:因为python的int类型字节数比其他语言的int字节数多, 因此移动次数将会很多,我这里使用python实现的时候发现陷入了死循环
陷入死循环的代码:
# 方法一的左移
class Solution(object):
def hammingWeight(self, n):
count = 0
temp = 1
while temp:
if n & temp:
count += 1
temp = temp << 1
return count
中间额外加了一个变量,循环超过32次则结束,还是有一定的缺陷
# 方法一的左移
class Solution(object):
def hammingWeight(self, n):
count = 0
temp = 1
t = 0
while temp:
t += 1
if n & temp:
count += 1
temp = temp << 1
if t >= 32:
break
return count
如下代码查看Python中的int的字节数
import sys
print(sys.getsizeof(5)) # 28
print(type(5)) #
// 剑指offer左移
int NumberOf(int n)
{
int count = 0;
unsigned int flag = 1;
while(flag)
{
if(n & flag)
count ++;
flag = flag << 1;
}
}
// 自己代码:方法一的左移
#include
int main()
{
int n = 4096;
int temp = 1;
int count = 0;
while (temp)
{
if (n & temp)
count += 1;
temp = temp << 1;
}
printf("%d\n", count);
return 0;
}
# 方法二
class Solution(object):
def hammingWeight(self, n):
count = 0
while n != 0:
count += 1
n &= (n - 1)
return count
# 方法三
class Solution(object):
def hammingWeight(self, n):
"""
:type n: int 要求二进制中1位数的数
:rtype: int 返回二进制中数字1的位数
"""
return bin(n).count('1')
其他类似题目
一条语句判断一个整数是否为2的整数次方
LeetCode231题
错误代码:
def is_two_pow(num):
# 因为一个数如果是2的整数次方,二进制表示只会存在一个1,因此num & (num - 1)将会返回一个0
if num & (num - 1):
return False
else:
return True
if __name__ == '__main__':
print(is_two_pow(3))
以上代码是错误的,没有考虑到num = 0的情况,因为0 & -1 也是为0
class Solution:
def isPowerOfTwo(self, n: int) -> bool:
if n == 0:
return False
return (False if n & (n - 1) else True)
输入两个整数m,n 计算需要改变m的二进制表示中的多少位才能得到n
def change_num(num1, num2):
temp = num1 ^ num2 # 得到异或的结果
count = 0 # 统计有多少位不同
while temp: # 此循环用于统计位1的个数
count += 1
temp = temp & (temp - 1)
print(count)
return count
if __name__ == '__main__':
change_num(4, 2)