【算法题】2 的 n 次幂的背后

本文介绍了如何利用位运算判断一个数是否为2的幂次,以及如何找到最近的2的幂次。通过二进制特性,可以高效地解决相关问题,如通过(x&(x-1))==0判断和找到距离给定数最近的2的幂次。此外,还探讨了如何通过加减2的幂次将数变为0的最少操作次数,并讨论了2的幂次与或运算的关系。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言: 说实话,真的不爱写算法题相关的文章了,觉得没啥意义,但是对这种比较好玩并且简单,学会了就能很好提高算法效率的文章,还是要写一写,不哭不哭,你不会不是你的错,只是因为你懒,学会了就好了,加油!!

一、2 的 n 次幂的特点

我们先看看 2 的 n 次幂都有什么 ?

n2 的 n 次幂二进制表示
1210
24100
381000
41610000

通过观察,我们可以得出,2 的 n 次幂的二进制表示只有最高位是 1, 其余位都是 0,这也是 2 的 n 次幂的特点 。



二、如何判断某个数 x 是否是 2 的 n 次幂

1. 暴力法

对 x 一直除 2 取余数 % , 看看最后当 x == 0 时,余数是否为 0 ,若是为 0 ,说明其是 2 的 n 次幂

2. 利用 2 的 n 次幂的二进制特点

2 的 n 次幂的二进制表示只有最高位为 1,其余位都为 0。因此,我们对其减 1 得到的结果的二进制表示只有原二进制最高位为 0, 其余位都为 1。如下表所示:

n2 的 n 次幂 (x)二进制表示(x-1) 的二进制表示x & (x-1)
1210010
241000110
38100001110
41610000011110
0

因此,我们将 该数字 x 按位与 (x - 1),若是结果为 0 ,说明该数字是 2 的 n 次幂

Java 代码表示如下:

        if((x & (x-1)) == 0){
            // 说明是 2 的幂次
            return true;
        }


三、如何找到距离某个数最近的 2 的 n 次幂

1. 暴力法

排除 , 我们就不使用暴力法,就不用,就不用 ~~

2. 重点:以数字的二进制形式进行思考

以数字 5 为例,距离它最近的2的幂次,比5大的有一个,比5小的有一个,5 的二进制表示为 101。

因为 2 的幂次只有最高位为 1,那么问题的关键就在于如何消除最低位的 1,消除的方法为 加上最低位 1xx(xx位置为 k 个 0) ,或者减去最低位的 1xx(xx位置为 k 个 0) 。

我们不断进行消除,最后就一定能得到距离 x 的最近的两个 2 的幂次

得到数字 x 的最低位 1xx(xx位置为 k 个 0) 的方法为 : x & (-x)

理解 x & (-x) 之前,我们先恶补一下计算机基础知识:
以 -5 为例:
1.  把这个负数的绝对值转换为二进制,即求原码 (1012.  把原码取反,即求反码 (111111111111111111111111111110103.  把反码加1,即求补码 (11111111111111111111111111111011)
所以, 5 & -5 = 1
解释: 通过上述变化 -5 只有最低位的 1xx(xx位置为 k 个 0)5相同,其余位都变得和 5 不同(即被取反了)

3. 寻找最近位置完整代码

    // 得到小于 n 的距离 n 最近的 2 的幂次
    public int getMinDistance(int n){

        // 直接就是 2 的 n 次幂
        if((n & (n - 1)) == 0){
            return n;
        }
        // 最低位的 1
        int lb = n & (-n);
        // 我们要通过不断加上减去最低位1,来消除低位的1,从而得到只有最高位为1的2的幂次
        // 但是我们要的是距离n最近的小于n的幂次,因此取两者的最小值
        return Math.min(getMinDistance(n + lb), getMinDistance(n - lb));
    }

    // 得到大于 n 的距离 n 最近的 2 的幂次
    public int getMaxDistance(int n){

        // 直接就是 2 的 n 次幂
        if((n & (n - 1)) == 0){
            return n;
        }
        // 最低位的 1
        int lb = n & (-n);
        // 我们要通过不断加上减去最低位1,来消除低位的1,从而得到只有最高位为1的2的幂次
        // 但是我们要的是距离n最近的大于n的幂次,因此取两者的最大值
        return Math.max(getMinDistance(n + lb), getMinDistance(n - lb));
    }

我们在对上面代码得到的结果,进行一个距离绝对值的判断,就能得到我们要的结果了

纠错: 需要这么绕嘛?
当然不需要,在实际实现时,我们只需要找到 x 的最高位 1 的位置,然后进行两次 1 的左移操作就行了。 (以 5 为例子, 1 << 2 (距离最近的小于 5 的位置),1 << 3 (距离最近的大于 5 的位置))

为了下一道题铺垫才这么做的 (试图为自己被绕进去了找借口,wwwww)



四、如何通过不断加减 2 的 n 次幂使得某个数最终结果为 0,最少操作次数

通过上面的分析,我们可以轻松得出这道综合题的完整代码,如下:

class Solution {
    // 6365. 将整数减少到零需要的最少操作数
    // 操作依据 —— 找到距离目标值最近的2的n次幂
    public int minOperations(int n) {
        return dfs(n);
    }

    public int dfs(int n){
        // 说明是 2 的幂次
        if((n & (n-1)) == 0){
            // 说明是 2 的幂次,最后再处理一次,即减去这个数字
            return 1;
        }
        int lb = n & (-n);
        // 操作次数 + 1,然后递归得到的新的数字
        return 1 + Math.min(dfs(n + lb), dfs(n - lb));
    }
}


五、2 的幂次与 或运算 的关系

1. 一个结论

(1)任何一个 非 2 的幂次的正整数 都能由 2 的幂次 通过按位或运算得到 , 即 x == k1 | k2 | k3 | ..,其中 x 为非 2 的幂次的正整数(例如 3, 5, 6 等),k1,k2 等为 2 的幂次
(2)2 的幂次不能通过或运算得到

解释:
2 的幂次的二进制表示为 1xxxxx (xxxxx位置为 k 个 0) ,那么通过多个不同的 2 的幂次进行或运算,我们可以让任意一位为 1 ,从而得到各种不同的正整数。同时也由于 2的幂次 只有最高位为 1, 所以它不能通过多个数字通过按位或运算得到 (除非其本身按位或 0 值)

2. 例题

给你一个下标从 0 开始的整数数组 nums 。

如果存在一些整数满足 0 <= index1 < index2 < ... < indexk < nums.length ,得到 nums[index1] | nums[index2] | ... | nums[indexk] = x ,那么我们说 x 是 可表达的 。换言之,如果一个整数能由 nums 的某个子序列的或运算得到,那么它就是可表达的。

请你返回 nums 不可表达的 最小非零整数 。

在这里插入图片描述

根据上面的分析,我们很容易写出本题的代码如下:

class Solution {
    // 优化,通过异或得到 11001111 然后取反得到 00110000,再得到最低位的1( 即1000),就是缺少的不能通过异或得到的最小数字
    public int minImpossibleOR(int[] nums) {
        int mask = 0;
        for (int num : nums){
            if(((num & (num - 1)) == 0)){ // 说明是 2 的位次
                mask |= num;
            }
        }
        mask = ~mask;
        return mask & -mask;
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值