位运算包括5种:与,或,异或,左移,右移
1、题目描述
输入一个整数,输出该数二进制表示中1的个数。其中负数用补码表示。
容易陷入死循环的写法:
class Solution {
public:
int NumberOf1(int n) {
int flag=1;
int cnt=0;
while(n){
if(n&flag!=0)
cnt++;
n=n>>1;
}
return cnt;
}
};
要考虑有符号数,右移时,左边补符号位,正数还好,补0,可负数补1,例如输入一个负数0x80000000,右移一位最高位补1是0xC0000000而不是0x40000000,多次移位后结果变成0xFFFFFFFF,陷入死循环。
所以,固定n,来左移flag:
class Solution {
public:
int NumberOf1(int n) {
unsigned int flag=1;
int cnt=0;
while(flag){
if(n&flag)
cnt++;
flag=flag<<1;
}
return cnt;
}
};
这个算法循环的次数等于整数的位数,32位整数需要循环32次
下面一个最优的方法是:(有多少个1就循环多少次)
用到一个结论:把一个整数减去1再与这个整数相与,会把整数最右边一个1变成0。例如:1100减去1得到1011,再将1100与1011做与运算,得到1000,最右边的1变成了0
所以有如下代码:
class Solution {
public:
int NumberOf1(int n) {
int cnt=0;
while(n){
n=(n-1)&n;
++cnt;
}
return cnt;
}
};
注意两个问题:
1.&比!=,==的优先级高
2、n>>1并不会改变n自身的结果,不是自运算
相关问题:
1、用一条语句判断一个整数是不是2的整数次方(肯定是正数)
一个数如果是2的整数次方,只包含一个1,用前面的方法,用这个整数与其减去1之后的结果相与就会得到0
2、输入两个整数m,n计算需要改变m的二进制表示中的多少位,才能得到n
异或的结果即为所求
更新:
3、不用加减乘除做两个数的加法
(1)异或,也就是不带进位的加法
(2)计算进位,只有1+1时产生进位得到10,所以直接相与,然后左移一位就是进位的结果
(3)将前两步结果相加
class Solution {
public:
int Add(int num1, int num2){
if((num1&num2)<<1)
return Add(num1^num2,(num1&num2)<<1);
else
return num1^num2;
}
};
4、不使用新的变量,交换两个变量的值
基于加减法的:a=a+b ;b=a-b;a=a-b;
基于异或运算:a=a^b;b=a^b;a=a^b;