LeetCode338——比特位计数

我的LeetCode代码仓:https://github.com/617076674/LeetCode

原题链接:https://leetcode-cn.com/problems/counting-bits/

题目描述:

LeetCode338——比特位计数_第1张图片

知识点:位运算、动态规划

思路一:动态规划

状态定义:f(x) -------- 数组中索引为x处的值

状态转移

(1)如果x是偶数,f(x) = f(x / 2)。因为对于偶数来说,其乘以2只是在所有位左移1位并在右边新添一个0,因此其1的个数不会发生变化。

(2)如果x是奇数,f(x) = f(x - 1)。奇数的1的个数一定和比它小的最大偶数的1的个数相同。

时间复杂度和空间复杂度均是O(n)。

JAVA代码:

public class Solution {

    public int[] countBits(int num) {
        if (num == 0) {
            return new int[] {0};
        } else if (num == 1) {
            return new int[] {0, 1};
        }
        int[] result = new int[num + 1];
        result[0] = 0;
        result[1] = 1;
        for (int i = 2; i < num + 1; i++) {
            if (i % 2 == 0) {
                result[i] = result[i / 2];
            } else {
                result[i] = result[i - 1] + 1;
            }
        }
        return result;
    }

}

LeetCode解题报告:

LeetCode338——比特位计数_第2张图片

思路二:位运算

i中1的个数一定比i & (i - 1)中1的个数多1个。

(1)如果i是奇数,显然i最右边的那位是1,而i & (i - 1)会消去最右边的那个1,而其余的1个数不变,1的总个数会减少1。

(2)如果i是偶数,显然i的最右边那位是0,而i & (i - 1)会使得最右边那位依然是0,而使得中间的某个1变为0。举个例子:

假设i = 00000000 00000000 00000000 10000000,那么

   i -1 = 00000000 00000000 00000000 01111111,显然i & (i - 1) = 00000000 00000000 00000000 00000000,1的总个数减少了1个。

时间复杂度和空间复杂度均是O(n)。

JAVA代码:

public class Solution {

    public int[] countBits(int num) {
        int[] result = new int[num + 1];
        result[0] = 0;
        for (int i = 1; i < num + 1; i++) {
            result[i] = result[i & (i - 1)] + 1;
        }
        return result;
    }

}

LeetCode解题报告:

LeetCode338——比特位计数_第3张图片

 

你可能感兴趣的:(LeetCode题解)