位运算相关题目

位运算包括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;

               



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值