牛客剑指offer刷题位运算篇

不用加减乘除做加法

题目

设计一个函数把两个数字相加。不得使用 + 或者其他算术运算符。

示例:
输入: a = 1, b = 1
输出: 2

提示:
a, b 均可能是负数或 0
结果不会溢出 32 位整数
LeetCode链接

思路

采用位运算思想:
a ^ b 的结果为 a+b 二进制运算不进位的情况;
a & b 的结果为 a+b 二进制运算进位的情况,当发生进位操作时,使用<< 1位即可;
通过不断 ^ & 操作判断是否仍然需要进位,直到不需要进位时,即可得到最终结果
LeetCode解题思路

代码实现
 public int add(int a, int b) {
        int m = a ^ b;  //用于表示相加结果
        int n = (a & b) << 1; //表示进位
        while(n != 0){
            int temp = m ^ n;
            n = (m & n) << 1;
            m = temp;
        }
        return m;
    }

二进制中1的个数

题目

编写一个函数,输入是一个无符号整数(以二进制串的形式),返回其二进制表达式中数字位数为 ‘1’ 的个数(也被称为 汉明重量).)。
LeetCode链接

思路

我们知道:
若 a & 1 = 1 说明a最后一位是1,如果为0,说明最后一位是0;
因此我们可以通过不断进行右移操作,判断a最后一位是否为1从而进行累加计算;
由于本题输出的是无符号整数,对应的应该使用无符号右移,对应 >>>

代码实现
 public int hammingWeight(int n) {
        int res = 0;
        while(n != 0){
            res += n & 1;
            n = n >>> 1;
        }
        return res;
    }

数值的整数次方

题目

描述
实现函数 double Power(double base, int exponent),求base的exponent次方。

注意:
1.保证base和exponent不同时为0。
2.不得使用库函数,同时不需要考虑大数问题
3.有特殊判题,不用考虑小数点后面0的位数。

示例1
输入:
2.00000,3
返回值:
8.00000

示例2
输入:
2.10000,3
返回值:
9.26100

示例3
输入:
2.00000,-2
返回值:
0.25000
说明:
2的-2次方等于1/4=0.25

牛客题目链接

思路

首先需要对exponent为负数的情况进行处理,也就对应(1/base)^(-exponent);
然后采用递归的思想,将计算base^exponent分解为求 base^(exponent/2) * base^(exponent - exponent/2),而exponent/2最终结果不是为0就是为1,即可以轻松计算出结果;

代码实现
 public double Power(double base, int exponent) {
 		//首先判断exponent是正数还是负数
        return exponent > 0 ? quickPower(base,exponent) :quickPower(1/base,-exponent); 
  }

  private double quickPower(double base, int exponent){
        if(exponent == 0){
            return 1;
        }
        if(exponent == 1){
            return base;
        }
        //递归求解
        return quickPower(base,exponent/2) * quickPower(base, exponent - exponent/2);
  }

数组中只出现一次的两个数字

题目

一个整型数组里除了两个数字只出现一次,其他的数字都出现了两次。请写程序找出这两个只出现一次的数字。

数据范围:数组长度 2≤n≤1000,数组中每个数的大小 0<val≤1000000
要求:空间复杂度 O(1),时间复杂度 O(n)

提示:输出时按非降序排列。
牛客题目链接

思路

采用位运算方式:
我们知道
a ^ 0 = a
a ^ a = 0
因此我们如下操作:

  1. 遍历数组,对所有数据进行^操作,得到结果为 res = a ^ b ;【后续肯定就想着如何拆分a和b】
  2. 使用k = 1和res进行&运算,通过对k不断<<1操作,找出 res中为1的那一位,这一位也表示a 和 b 不同的一位【因为是异或运算,不同才为1】;
  3. 再次遍历数组,通过数字 & k 是否为 0 来对数组进行分组,在对分组的所有数据进行异或运算,同组中的相同数字必然会得到0,这样就可以得出 a 和 b数据;
    牛客大神解题思路
代码实现
 public int[] FindNumsAppearOnce (int[] nums) {
       if(nums == null || nums.length < 2){
            return new int[2];
       }
       int res = 0;
       int num1 = 0;
       int num2 = 0;
       for(int num : nums){
         res ^= num;
       }
       int k = 1;
       while((k & res) == 0){
            k <<= 1;
       }
       for(int num : nums){
            if((k & num) == 0){
                num1 ^=num;
            }else{
                num2 ^=num;
            }
       }
       return num1 < num2 ? new int[]{num1,num2} : new int[]{num2,num1};
    }

求1+2+3+…+n

题目

求1+2+3+…+n,要求不能使用乘除法、for、while、if、else、switch、case等关键字及条件判断语句(A?B:C)。

数据范围: 0<n≤200
进阶: 空间复杂度 O(1) ,时间复杂度 O(n)
牛客题目链接

思路

由于题目的一些限制,我们想到可以通过递归方式解决问题:
求n的和,即为求 (1+…n-1) + n的和;
求n-1的和,即为求 (1+…n-2) + n-1的和;

求2的和,即为求(0 + 1) + 2的和;
同时又不能使用条件判断方式结束递归,因此我们使用&&运算:

在函数中,如果与运算成立,则继续,否则终止函数直接返回false。

代码实现
public int Sum_Solution(int n) {
        boolean flag = (n > 1) && ((n+=Sum_Solution(n-1)) > 0 );
        return n;
    }
  • 9
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值