【优选算法篇】位运算小课堂:从入门到精通的奇妙之旅(上篇)

文章目录

须知

💬 欢迎讨论:如果你在学习过程中有任何问题或想法,欢迎在评论区留言,我们一起交流学习。你的支持是我继续创作的动力!

👍 点赞、收藏与分享:觉得这篇文章对你有帮助吗?别忘了点赞、收藏并分享给更多的小伙伴哦!你们的支持是我不断进步的动力!
🚀 分享给更多人:如果你觉得这篇文章对你有帮助,欢迎分享给更多对C++算法感兴趣的朋友,让我们一起进步!

1. C++ 位运算算法 详解

1.1 位运算的重要性

位运算(Bitwise Operation)是计算机底层操作的核心。它的重要性体现在以下几个方面:

  1. 高效性:位运算直接操作二进制位,速度极快,是最高效的运算方式之一。
  2. 底层控制:用于硬件控制、网络协议、图像处理等领域,帮助开发者操作底层数据。
  3. 空间优化:在数据压缩、哈希算法等场景中,通过位操作极大节省存储空间。
  4. 算法核心:许多高效算法依赖位运算实现,比如快速幂、判断奇偶、位图等。

1.2 位运算的定义

位运算是对整数在二进制位层面进行操作的运算,主要包括以下基本操作:

  1. 按位与(&)

    • 规则:两个二进制位都为 1 时,结果为 1;否则为 0
    • 应用:常用于掩码操作,保留某些位的信息。
      • 示例:1101 & 1011 = 1001
  2. 按位或(|)

    • 规则:只要有一个二进制位为 1,结果为 1;否则为 0
    • 应用:设置某些位为 1
      • 示例:1101 | 1011 = 1111
  3. 按位异或(^)

    • 规则:二进制位相同为 0,不同为 1
    • 应用:交换值、检测位差。
      • 示例:1101 ^ 1011 = 0110
  4. 按位取反(~)

    • 规则:将每个位取反,即 1001
    • 应用:构造补码。
      • 示例:~1101 = 0010(假设为4位操作)
  5. 左移(<<)

    • 规则:将所有位向左移动指定位置,右边补 0
    • 应用:高效计算乘以 2^n
      • 示例:1101 << 2 = 110100
  6. 右移(>>)

    • 规则:将所有位向右移动指定位置。
      • 算术右移:高位用符号位填充,保留符号。
      • 逻辑右移:高位用 0 填充。
    • 应用:高效计算除以 2^n
      • 示例:1101 >> 2 = 0011

1.3 位运算的核心思想

  1. 二进制数据的直接操作:将复杂的逻辑转换为简单的二进制运算。
  2. 高效替代算术操作:使用移位、与或操作完成加减乘除或逻辑控制。
  3. 灵活的位级处理:针对数据的某些特定位进行精确控制和修改。
  4. 组合运用:通过多种位运算的结合,构造高效的算法。

1.4 经典应用

  1. 快速判断奇偶数

    • 思路:通过按位与操作 n & 1 判断奇偶。
      • 示例:5 & 1 = 1(奇数),4 & 1 = 0(偶数)
  2. 交换两个数(不使用额外变量)

    a = a ^ b b = a ^ b a = a ^ b

    • 示例:a = 5, b = 7 -> a = 7, b = 5
  3. 清零最低位的 1

    • 公式n & (n - 1)
    • 作用:用于统计 1 的个数。
      • 示例:12 (1100) -> n & (n - 1) = 8 (1000)
  4. 判断二进制中是否为 2 的幂

    • 公式n > 0 and (n & (n - 1)) == 0
      • 示例:8 (1000)2 的幂;10 (1010) 不是。
  5. 位图(Bitmap)算法

    • 应用于海量数据处理,用位表示某个值是否存在,大幅节省空间。
      • 示例:检测某个数字是否出现。
  6. 位掩码(Bitmask)操作

    • 用于权限管理(如读写执行权限),每个位表示一个权限状态。
      • 示例:111 表示 rwx 权限,101 表示 rw-
  7. 图像处理中的像素操作

    • 按位与、或操作对图像的颜色、透明度等位级数据进行精细控制。
  8. 加法的位运算实现

    • 利用异或操作模拟加法:a ^ b(不进位加法)和 a & b(进位)。
  9. 哈希函数优化

    • 通过位运算快速计算哈希值或减少冲突。
  10. 网络编程中的子网掩码

  • 使用按位与确定 IP 地址和子网的关系。

位运算的基础简单但应用广泛,掌握它不仅能提升代码效率,还能帮助开发者深入理解计算机的工作原理。

 2. 题目1:位1的个数

题目链接:191. 位1的个数 - 力扣(LeetCode)

题目描述:

 2.1 算法思路:

算法核心思想

  1. 逐位检查

    • 利用右移(>>)操作,将 n 的最低位逐步移出,然后与 1 按位与(&)操作来检查这一位是否为 1
    • 如果结果是 1,说明当前位是 1,计数器 count 增加 1
  2. 循环 32

    • 因为输入是一个 32 位整数(int),需要从最低位到最高位依次检查所有 32 位。
  3. 返回计数器

    • 每次遇到 1 时,计数器增加,最终返回这个计数值。

2.2 代码详解

class Solution 
{
public:
    int hammingWeight(int n) 
    {
        int count = 0; // 初始化计数器,统计 1 的个数
        for (int i = 0; i < 32; i++) // 遍历 32 位
        {
            if (((n >> i) & 1) == 1) // 右移 i 位后,与 1 按位与,判断最低位是否为 1
                count++; // 如果是 1,增加计数
        }
        return count; // 返回汉明重量
    }
};
2.2.1 示例分析

输入:n = 11

二进制表示:00000000000000000000000000001011
逐步过程如下:

  • i = 0: (n >> 0) & 1 = 000...0011 & 1 = 1 -> count = 1
  • i = 1: (n >> 1) & 1 = 000...001 & 1 = 1 -> count = 2
  • i = 2: (n >> 2) & 1 = 000...000 & 1 = 0 -> count = 2
  • i = 3: (n >> 3) & 1 = 000...000 & 1 = 1 -> count = 3
    最终结果:count = 3

2.3 时间复杂度与空间复杂度

  1. 时间复杂度

    • 循环固定执行 32 次(因为是 32 位整数)。
    • 每次循环中执行常数操作:右移和按位与。
    • 总时间复杂度为 O(1),与输入值大小无关。
  2. 空间复杂度

    • 只使用了一个额外变量 count,因此是 O(1)

2.4 改进优化

虽然代码的时间复杂度已经是 O(1),但在某些场景下可以进一步优化位操作的效率,例如使用 n & (n - 1) 清零最低位的 1
优化后的代码:

class Solution 
{
public:
    int hammingWeight(int n) 
    {
        int count = 0;
        while (n != 0) // 当 n 不为 0 时,逐位清零最低位的 1
        {
            n &= (n - 1); // 清零最低位的 1
            count++;
        }
        return count;
    }
};
2.4.1 优化思路:
  • 每次操作将当前数字的最低位 1 清零,使得循环次数仅等于 1 的个数。
  • 时间复杂度O(k),其中 k1 的个数,适用于输入值中 1 的个数远小于 32 的情况。

2.5 总结

  • 原始方法适合简单直接的逐位统计,适合入门和通用情况。
  • 优化方法更高效,尤其在实际应用中,适用于稀疏数据(1 的个数较少)的场景。

3. 题目2:比特位计数

题目链接:338. 比特位计数 - 力扣(LeetCode) 

题目描述:

3.1 算法思路:

  1. 辅助函数 hammingWeight 的功能

    • 作用:输入一个整数 n,计算其二进制中 1 的个数。
    • 思路:通过循环,逐位检查 n 的每一位是否为 1,如果是,则计数器增加。
    • 实现细节
      • 使用右移运算符 >> 将数字逐位右移。
      • 1 进行按位与(&)运算,检查最低位是否为 1
      • 循环 32 次(针对 32 位整数)。
  2. 主函数 countBits 的功能

    • 作用:从 0n,依次调用 hammingWeight 计算每个数字中 1 的个数,并将结果存入数组 arr
    • 思路
      • 遍历所有数字 i,范围是 [0, n]
      • 对每个数字 i 调用 hammingWeight,获取其二进制中 1 的个数。
      • 使用 arr.push_back(count) 将结果依次存入数组。
  3. 整体流程

    • 初始化数组 arr 用于存储结果。
    • 循环从 0n,计算每个数字的 1 个数。
    • 返回结果数组 arr

3.2 示例代码:

// 定义一个辅助函数,用于计算一个整数的二进制中 '1' 的个数
int hammingWeight(int n) 
{
    int count = 0; // 初始化计数器,用于统计二进制中 '1' 的数量
    for (int i = 0; i < 32; i++) // 遍历整数的 32 位(假设输入是 32 位整数)
    {
        // 右移 i 位,并用按位与操作提取最低位,检查是否为 1
        if (((n >> i) & 1) == 1) 
            count++; // 如果当前位是 1,计数器加 1
    }
    return count; // 返回 '1' 的总个数
}

// 定义一个类,包含主函数,用于生成从 0 到 n 每个数字的二进制中 '1' 的个数
class Solution 
{
public:
    // 主函数,生成结果数组
    vector<int> countBits(int n) 
    {
        vector<int> arr; // 定义一个向量用于存储结果,每个元素代表对应数字的二进制中 '1' 的个数
        int count1 = 0; // 临时变量,用于存储每个数字的 '1' 个数

        // 遍历从 0 到 n 的每个数字
        for (int i = 0; i <= n; i++) 
        {
            count1 = hammingWeight(i); // 调用辅助函数,计算当前数字的 '1' 个数
            arr.push_back(count1); // 将结果添加到结果数组中
        }

        return arr; // 返回结果数组
    }
};
3.2.1 代码详解

辅助函数 hammingWeight

int hammingWeight(int n) 
{
    int count = 0;
    for (int i = 0; i < 32; i++) // 遍历整数的 32 位
    {
        if (((n >> i) & 1) == 1) // 检查当前位是否为 1
            count++; // 如果是,增加计数
    }
    return count; // 返回 1 的个数
}

 主函数 countBits

class Solution 
{
public:
    vector<int> countBits(int n) 
    {
        vector<int> arr; // 用于存储每个数字的汉明重量
        int count1 = 0;
        for (int i = 0; i <= n; i++) // 遍历从 0 到 n 的每个数字
        {
            count1 = hammingWeight(i); // 计算当前数字的二进制中 1 的个数
            arr.push_back(count1); // 将结果加入数组
        }
        return arr; // 返回结果数组
    }
};

示例分析

示例输入:n = 5

目标:生成从 05 的二进制中 1 的个数,即:

  • 0 -> 00001 的个数 = 0
  • 1 -> 00011 的个数 = 1
  • 2 -> 00101 的个数 = 1
  • 3 -> 00111 的个数 = 2
  • 4 -> 01001 的个数 = 1
  • 5 -> 01011 的个数 = 2

输出:[0, 1, 1, 2, 1, 2]

逐步过程:

  1. i = 0 → 调用 hammingWeight(0) → 结果 0arr = [0]
  2. i = 1 → 调用 hammingWeight(1) → 结果 1arr = [0, 1]
  3. i = 2 → 调用 hammingWeight(2) → 结果 1 arr = [0, 1, 1]
  4. i = 3 → 调用 hammingWeight(3) → 结果 2arr = [0, 1, 1, 2]
  5. i = 4 → 调用 hammingWeight(4) → 结果 1 arr = [0, 1, 1, 2, 1]
  6. i = 5 → 调用 hammingWeight(5) → 结果 2arr = [0, 1, 1, 2, 1, 2]

最终返回 arr = [0, 1, 1, 2, 1, 2]


3.3 时间复杂度与空间复杂度

时间复杂度

  1. 主函数:从 0n 遍历了 n + 1 个数字。
  2. 辅助函数:每次调用 hammingWeight 执行了 32 次操作(检查 32 位)。
  3. 总体复杂度O(n * 32),即 O(32n),等价于 O(n)

空间复杂度

  • 存储结果数组 arr,大小为 n + 1
  • 总空间复杂度为 O(n)

3.4 优化思路

当前实现中,每次计算每个数字的汉明重量时,完全独立地逐位检查。可以通过 动态规划 优化,利用前一个数字的结果推导当前数字的汉明重量。优化方法为:

  1. 观察规律
    • 如果 i 是偶数,则 countBits(i) = countBits(i / 2)(右移一位,相当于去掉最低位)。
    • 如果 i 是奇数,则 countBits(i) = countBits(i / 2) + 1(最低位为 1)。
  2. 动态规划实现
class Solution 
{
public:
    vector<int> countBits(int n) 
    {
        vector<int> dp(n + 1, 0); // 初始化结果数组,dp[i] 表示数字 i 的汉明重量
        for (int i = 1; i <= n; i++) 
        {
            dp[i] = dp[i >> 1] + (i & 1); // 动态转移方程
        }
        return dp;
    }
};
  1. 优化复杂度:时间复杂度降为 O(n),空间复杂度仍为 O(n)

3.5 总结

  • 初始实现通过逐位计算每个数字的汉明重量,简单直观,时间复杂度为 O(n * 32)
  • 动态规划优化利用子问题结果,减少重复计算,时间复杂度降为 O(n)

4. 题目3:汉明距离

题目链接:461. 汉明距离 - 力扣(LeetCode)

题目描述:

4.1 算法思路:

算法分为两部分

  1. 辅助函数 hammingWeight

    • 计算一个整数的二进制表示中 1 的个数(即汉明重量,Hamming Weight)。
    • 通过逐位右移(>>)和按位与(&)操作,统计 1 的个数。
  2. 主函数 hammingDistance

    • 计算两个整数 xy 的汉明距离。
    • 使用按位异或(^)操作找出 xy 的二进制表示中不同的位,结果是一个新整数 s
    • 调用 hammingWeight 函数统计 s 中的 1 的个数,即为汉明距离。

 4.2 示例代码:

// 辅助函数:计算整数 n 的二进制表示中 '1' 的个数
int hammingWeight(int n) 
{
    int count = 0; // 初始化计数器,用于统计二进制中 '1' 的数量
    for (int i = 0; i < 32; i++) // 遍历 32 位(假设输入为 32 位整数)
    {
        // 右移 i 位,并与 1 进行按位与操作,检查最低位是否为 1
        if (((n >> i) & 1) == 1) 
            count++; // 如果最低位为 1,则计数器加 1
    }
    return count; // 返回二进制中 '1' 的总数
}

// 主类:用于计算两个整数的汉明距离
class Solution 
{
public:
    // 主函数:计算 x 和 y 的汉明距离
    int hammingDistance(int x, int y) 
    {
        // 按位异或操作,得到二进制中 x 和 y 不同的位
        // 异或规则:相同位结果为 0,不同位结果为 1
        int s = x ^ y;

        // 调用辅助函数 hammingWeight,统计异或结果中 '1' 的个数
        // 这些 '1' 的数量即为 x 和 y 的汉明距离
        return hammingWeight(s);
    }
};
4.2.1 代码逻辑详细解析
  1. 辅助函数 hammingWeight

    • 功能:统计整数 n 的二进制表示中 1 的个数。
    • 方法:通过逐位右移操作,依次提取最低位并判断是否为 1
      • n >> i:将整数 n 右移 i 位。
      • (n >> i) & 1:提取右移后结果的最低位,并判断是否为 1
    • 时间复杂度:固定遍历 32 位,时间复杂度为 O(32),即 O(1)
  2. 主函数 hammingDistance

    • 功能:计算两个整数 xy 的汉明距离。
    • 方法
      • 通过按位异或(x ^ y)操作,生成一个新的整数 s,其二进制表示中 1 的位置对应 xy 不同的位。
        • 异或规则:

          1 ^ 1 = 0 0 ^ 0 = 0 1 ^ 0 = 1 0 ^ 1 = 1

        • 因此,s 的二进制中所有的 1 表示 xy 不同的位。
      • 调用 hammingWeight 函数,统计 s1 的个数,作为 xy 的汉明距离。
    • 时间复杂度
      • 异或操作时间复杂度为 O(1)
      • 调用 hammingWeight 时间复杂度为 O(32),即 O(1)

示例运行过程

示例输入

x = 1, y = 4 

计算过程

  1. x 的二进制表示:0001
    y 的二进制表示:0100
    按位异或:

x ^ y = 0001 ^ 0100 = 0101

  • 异或结果为:0101

  • 调用 hammingWeight(5)

    • 5 的二进制表示为:0101
    • 遍历每一位:
      • 0 位:1count = 1
      • 1 位:0count = 1
      • 2 位:1count = 2
      • 3 位:0count = 2
    • 返回 count = 2
  • 最终返回汉明距离:2

输出

4.3 时间复杂度和空间复杂度

时间复杂度

  1. 异或操作:按位异或操作的时间复杂度为 O(1)
  2. 辅助函数 hammingWeight:遍历 32 位,时间复杂度为 O(1)
  3. 总时间复杂度O(1)

空间复杂度

  • 使用常量级的变量(如 counts),不需要额外空间,O(1)

4.4 代码特点

  1. 优点

    • 简单直观,通过辅助函数分离逻辑,代码结构清晰。
    • 时间复杂度和空间复杂度均为常量级,性能优良。
  2. 可优化方向

    • 优化 hammingWeight 函数,利用 n & (n - 1) 方法快速清除最低位的 1,减少迭代次数

优化后的辅助函数

int hammingWeight(int n) 
{
    int count = 0;
    while (n != 0) // 当 n 不为 0 时
    {
        n &= (n - 1); // 清除最低位的 1
        count++; // 每次清除后计数加 1
    }
    return count; // 返回二进制中 '1' 的总数
}
  • 优势:循环次数等于 1 的个数,适用于二进制中 1 较少的情况,进一步提升性能。

完整优化代码

int hammingWeight(int n) 
{
    int count = 0;
    while (n != 0) 
    {
        n &= (n - 1);
        count++;
    }
    return count;
}

class Solution 
{
public:
    int hammingDistance(int x, int y) 
    {
        int s = x ^ y; // 按位异或,找出 x 和 y 不同的位
        return hammingWeight(s); // 统计不同位的数量(即 '1' 的个数)
    }
};

 5. 题目4:只出现1的次数

题目链接:136. 只出现一次的数字 - 力扣(LeetCode)

题目描述:

 5.1 算法思路

  1. 异或运算的特性

    • 异或(^)运算的规则:
      • a ^ a = 0(任何数与自己异或,结果为 0
      • a ^ 0 = a(任何数与 0 异或,结果不变)
      • 异或满足交换律和结合律a ^ b ^ c = a ^ c ^ b
    • 因此,如果一个数出现两次,它们会相互抵消为 0
    • 只出现一次的数与 0 异或后仍是其本身。
  2. 核心逻辑

    • 遍历数组中的每个元素,对每个元素执行 异或操作
    • 最终,所有出现两次的数字会抵消为 0,只剩下那个出现一次的数字。

5.2 示例代码:

class Solution 
{
public:
    int singleNumber(vector<int>& nums) 
    {
        int ret = 0; // 初始化结果为 0
        for (auto s : nums) // 遍历数组中的每个元素
        {
            ret ^= s; // 异或操作:逐个元素与结果异或
        }
        return ret; // 返回出现一次的数字
    }
};
5.2.1 执行过程示例

输入示例

nums = [4, 1, 2, 1, 2]

执行过程

最终结果:4 

5.3 时间复杂度与空间复杂度

  1. 时间复杂度

    • 遍历整个数组一次,每个元素进行一次异或操作。
    • 时间复杂度为 O(n),其中 n 是数组的长度。
  2. 空间复杂度

    • 使用了一个整数 ret 进行累加异或,不需要额外的数据结构。
    • 空间复杂度为 O(1)

5.4 多种解法

5.4.1 解法二:哈希表法

思路

  1. 使用 哈希表 记录每个元素出现的次数。
  2. 遍历哈希表,找到次数为1的元素。

代码实现

class Solution {
public:
    int singleNumber(vector<int>& nums) {
        unordered_map<int, int> countMap;
        for (int num : nums) {
            countMap[num]++; // 记录每个元素出现的次数
        }
        for (auto& pair : countMap) {
            if (pair.second == 1) { // 找到只出现一次的元素
                return pair.first;
            }
        }
        return -1; // 兜底返回
    }
};

时间复杂度O(n) — 遍历数组和哈希表

空间复杂度O(n) — 额外使用哈希表存储元素及其频次


5.4.2 解法三:排序法

思路

  1. 对数组进行排序,相同的元素会相邻。
  2. 遍历排序后的数组,如果某个元素与前一个和后一个元素不同,它就是只出现一次的数。

代码实现

class Solution {
public:
    int singleNumber(vector<int>& nums) {
        sort(nums.begin(), nums.end()); // 排序数组
        for (int i = 0; i < nums.size(); i += 2) { // 每次跳两个位置
            if (i == nums.size() - 1 || nums[i] != nums[i + 1]) { 
                return nums[i]; // 如果元素和下一个元素不同,返回该元素
            }
        }
        return -1; // 兜底返回
    }
};

时间复杂度O(n log n) — 排序的时间复杂度

空间复杂度O(1) — 如果排序算法为原地排序


5.4.3 解法四:数学法(2∗sum - sum

思路

  1. 利用集合(Set)去重后计算数组中元素的和。
  2. 原始数组中所有元素之和减去去重后的和的两倍,结果就是只出现一次的数。

代码实现

class Solution {
public:
    int singleNumber(vector<int>& nums) {
        unordered_set<int> numSet;
        int sumOfNums = 0, sumOfSet = 0;

        for (int num : nums) {
            sumOfNums += num; // 数组中所有元素的和
            if (numSet.find(num) == numSet.end()) { // 如果元素不在集合中
                numSet.insert(num);
                sumOfSet += num; // 记录集合中元素的和
            }
        }
        return 2 * sumOfSet - sumOfNums; // 根据公式计算结果
    }
};

时间复杂度:O(n) — 遍历数组

空间复杂度:O(n) — 额外使用集合


5.4.4  解法对比总结

5.5 总结

  • 关键点:利用异或的特性,重复出现的数字会被抵消(a ^ a = 0),最终只剩下不重复的那个数。
  • 优势
    • 时间复杂度为 O(n),一次遍历解决问题。
    • 空间复杂度为 O(1),无需额外存储空间。
  • 适用场景:数组中有且只有一个数字出现一次,其余数字均出现两次

 6. 题目5:只出现一次的数字 |||

题目链接:260. 只出现一次的数字 III - 力扣(LeetCode)

题目描述:

 6.1 算法思路:

思路解析

步骤1:使用异或找出两个数字的异或结果

  1. 将数组中所有数字进行异或,最终结果 xorsum 是这两个不同数字的异或结果。
    • 原因
      • 相同的数字异或为 0a ^ a = 0)。
      • 0 与任何数异或不改变数值(a ^ 0 = a)。
      • 因此,数组中成对的数字会相互抵消,只剩下两个不同的数字的异或结果。
    • xorsum 的特点:
      • 它是两个只出现一次的数字的 异或值,即 xorsum = num1 ^ num2,其中 num1num2 是结果。

步骤2:找到异或结果中最低的不同位(区分两组数字)

  1. 异或结果 xorsum 中的 最低位的1,表示在这一位上,两个不同数字的二进制表示不同(一个是 1,另一个是 0)。

    • 通过 xorsum & (-xorsum) 找出最低的1位:
      • (-xorsum)xorsum补码,补码中保留最低位的1并将其他位取反。
      • xorsum & (-xorsum) 可以快速找到最低位的1
  2. 这位 lsblowest significant bit)可以将数组中的所有数字分为两组:

    • 第一组:在该位上为 1 的数字。
    • 第二组:在该位上为 0 的数字。

步骤3:分别异或两组数字,得到两个结果

  1. 将数组中的所有数字根据最低位的1进行分组。
  2. 对每一组的数字分别进行异或:
    • 在第一组中,所有成对的数字(出现两次的数字)会互相抵消,最终结果是 num1
    • 在第二组中,所有成对的数字也会互相抵消,最终结果是 num2

6.2 代码解析

class Solution {
public:
    vector<int> singleNumber(vector<int>& nums) {
        int xorsum = 0; // 用于存储所有数字的异或结果
        for (int num: nums) {
            xorsum ^= num; // 所有数字异或,最终结果是两个不同数字的异或值
        }

        // 找出 xorsum 中最低位的 1(用于区分两个数字)
        // 如果 xorsum 是 INT_MIN(最小负数),它本身只有最高位为 1,不会溢出
        int lsb = (xorsum == INT_MIN ? xorsum : xorsum & (-xorsum));

        int type1 = 0, type2 = 0; // 用于存储两个只出现一次的数字
        for (int num: nums) {
            if (num & lsb) { // 根据最低位的1分组:该位为1
                type1 ^= num; // 第一组异或,得到 num1
            }
            else { // 该位为0
                type2 ^= num; // 第二组异或,得到 num2
            }
        }

        return {type1, type2}; // 返回结果
    }
};
6.2.1 示例执行过程

输入

nums = [1, 2, 1, 3, 2, 5]

步骤1:计算所有元素的异或结果

  • 1 ^ 2 ^ 1 ^ 3 ^ 2 ^ 5 = 3 ^ 5 = 6(二进制 0110
  • 635 的异或结果。

步骤2:找到最低位的1

  • xorsum = 6 → 二进制 0110
  • xorsum & (-xorsum) = 2(最低位的1在第2位)。

步骤3:根据最低位的1分组

  • 分组1(第2位为1的数字):2, 2, 3
  • 分组2(第2位为0的数字):1, 1, 5

步骤4:分别异或两组

  • 第一组:2 ^ 2 ^ 3 = 3
  • 第二组:1 ^ 1 ^ 5 = 5

输出

[3, 5](结果顺序不唯一)。


6.3 复杂度分析 

时间复杂度分析

  1. 第一次遍历:计算所有数字的异或,时间复杂度为 O(n)

  2. 第二次遍历:根据最低位的1分组并分别异或,时间复杂度为 O(n)

  3. 总体时间复杂度:O(n)

空间复杂度分析

  • 使用了常数额外空间,空间复杂度为 O(1)

6.4  多种解法思路

6.4.1 解法二:哈希表法

思路

  1. 使用哈希表记录每个元素的出现次数。
  2. 遍历哈希表,找到出现次数为1的两个元素。

代码实现

class Solution {
public:
    vector<int> singleNumber(vector<int>& nums) {
        unordered_map<int, int> freq; // 统计元素出现的频率
        for (int num : nums) {
            freq[num]++;
        }
        
        vector<int> result;
        for (auto& pair : freq) {
            if (pair.second == 1) { // 出现次数为1的元素
                result.push_back(pair.first);
            }
        }
        
        return result;
    }
};

复杂度分析

  • 时间复杂度O(n),一次遍历数组和哈希表。
  • 空间复杂度O(n),额外的哈希表空间。

6.4.2 解法三:排序法

思路

  1. 对数组进行排序,相同的元素会相邻。
  2. 遍历排序后的数组,如果某个元素与前后元素都不相同,则它是目标元素之一。

代码实现

class Solution {
public:
    vector<int> singleNumber(vector<int>& nums) {
        sort(nums.begin(), nums.end()); // 排序数组
        vector<int> result;
        
        for (int i = 0; i < nums.size(); i++) {
            if ((i == 0 || nums[i] != nums[i - 1]) && 
                (i == nums.size() - 1 || nums[i] != nums[i + 1])) {
                result.push_back(nums[i]);
            }
        }
        
        return result;
    }
};

复杂度分析

  • 时间复杂度O(n log n),排序的时间复杂度。
  • 空间复杂度O(1),原地排序(如果排序算法是原地的)
 6.4.3 解法对比总结

6.5 总结

  1. 异或法是解决此类问题的核心思想,利用异或的性质实现高效解法。
  2. 通过 最低位的1 将两个不同的数字区分开来,巧妙地将问题分解为两组。
  3. 该解法时间复杂度为 O(n),空间复杂度为 O(1),性能最优。

 7. 总结要点

  1. 异或运算 是位运算解决重复出现问题的核心思想:
    • 相同为00与任何数异或等于数本身。
  2. 位统计 + 取模:通过逐位统计 1 的个数,可以解决 K次出现问题
  3. 低位的1:通过提取最低位的1,可以将数字分组,解决两个目标数字的问题。

8. 最后

 通过上面几个例题:「Single Number」的异或运算解法「Single Number III」的异或+分组方法,以及**「Single Number II」的位统计与取模优化**,我们总结出位运算在数组问题中的高效应用。

位运算 通过将复杂的数字出现次数、分组、去重等问题转化为简单的位级操作(如异或位统计取模等),极大地提升了问题求解的效率。
在处理 元素重复出现问题分组求解问题 以及 高效数值处理 等场景时,位运算展现出显著的性能优势,尤其适用于 大规模数据时间复杂度敏感 的应用场景。

路虽远,行则将至;事虽难,做则必成 

亲爱的读者们,下一篇文章再会!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/938200.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

pytest入门九:feature

fixture是pytest特有的功能&#xff0c;用以在测试执行前和执行后进行必要的准备和清理工作。使用pytest.fixture标识&#xff0c;定义在函数前面。在你编写测试函数的时候&#xff0c;你可以将此函数名称做为传入参数&#xff0c;pytest将会以依赖注入方式&#xff0c;将该函数…

Day9 神经网络的偏导数基础

多变量函数与神经网络 在神经网络中&#xff0c;我们经常遇到多变量函数。这些函数通常描述了网络的输入、权重、偏置与输出之间的关系。例如&#xff0c;一个简单的神经元输出可以表示为&#xff1a; z f ( w 1 x 1 w 2 x 2 … w n x n b ) z f(w_1x_1 w_2x_2 \ldots…

sg-exam:Star 2.2k,一套完善的在线教育平台,支持在线考试、在线学习,教育项目用它就没有错~~

​嗨&#xff0c;大家好&#xff0c;我是小华同学&#xff0c;关注我们获得“最新、最全、最优质”开源项目和高效工作学习方法 sg-exam是一个基于Java语言的在线考试系统&#xff0c;它集成了试卷管理、试题管理、考试安排、在线作答、自动阅卷等功能。该项目旨在帮助教育机构…

ArkTS中string和String/number和Number类型大小写的区别

ArkTS和TypeScript类似&#xff0c;string 和 String&#xff0c;number 和 Number 之间有一些重要的区别&#xff1a; 基本类型与对象类型 基本类型 (string, number)&#xff1a; string 和 number 是基本数据类型&#xff0c;用于表示原始值。例如&#xff1a;let str: str…

Ubuntu22.04切换gcc版本教程

在编译安装程序的时候,由于gcc版本过高,导致编译无法通过,需要降低gcc版本。 一、安装gcc版本 根据自己的需求安装gcc版本。 sudo apt update sudo apt install gcc-10 g++-10二、切换gcc版本 sudo update-alternatives --install /usr/bin/gcc gcc

c++领域展开第四幕——类和对象(上篇收尾 this指针、c++和c语言的初步对比)超详细!!!!

文章目录 前言一、this指针二、c和c语言的初步对比总结 前言 上篇我们初步学习了类的基本概念以及实例化 今天我们来学习类的构造以及析构还有类的默认成员函数&#xff0c;类和对象这一部分都会有点难 跟着我一起来吧 一、this指针 Date类中有 Init 与 Print 两个成员函数&…

python | linux | ModuleNotFoundError: No module named ‘WFlib‘ |找不到模块

问题&#xff1a; (base) beautyby521-7:~/Website-Fingerprinting-Library-master$ bash scripts/NetCLR.sh Traceback (most recent call last):File "/home/beauty/Website-Fingerprinting-Library-master/exp/pretrain.py", line 8, in <module>from WFli…

联发科MTK8788_MT8788安卓核心板安兔兔跑分_安卓主板方案商

MT8788安卓核心板具有集成的蓝牙、fm、WLAN和gps模块&#xff0c;是一个高度集成的基带平台&#xff0c;包括调制解调器和应用处理子系统&#xff0c;启用LTE/LTE-A和C2K智能设备应用程序。该芯片集成了工作在2.0GHz的ARM Cortex-A73、最高可达2.0GHz的ARM Cortex-A53和功能强大…

云计算HCIP-OpenStack02

书接上回&#xff1a; 云计算HCIP-OpenStack01-CSDN博客 7.OpenStack核心服务 7.1Horizon&#xff1a;界面管理服务 Horizon提供了OpenStack中基于web界面的管理控制页面&#xff0c;用户或者是管理员都需要通过该服务进行OpenStack的访问和控制 界面管理服务需要依赖于keyston…

ElasticSearch的自动补全功能(拼音分词器、自定义分词器、DSL实现自动补全查询、RestAPI实现自动补全查询)

文章目录 1. 什么是自动补全2. 拼音分词器2.1 初识拼音分词器2.2 下载拼音分词器2.3 安装拼音分词器2.4 测试拼音分词器 3. 自定义分词器3.1 拼音分词器存在的问题3.2 分词器&#xff08;analyzer&#xff09;的组成3.3 如何自定义分词器3.4 拼音分词器的可选参数3.5 配置自定义…

AI一键分析小红书对标账号‼️

宝子们&#xff0c;AI小助手近期发现了一款宝藏AI工具&#xff0c;拥有对标账号AI分析功能&#xff0c;只需10秒就能全面掌握对标账号的运营情况&#xff0c;并且可以根据分析结果提供创作方向和灵感&#xff0c;轻松助力1:1复刻起号&#xff01; 功能亮点&#xff1a; &…

【5G】5G的主要架构选项

最初&#xff0c;在3GPP讨论中考虑了所有可能的聚合和核心网络组合&#xff0c;共有八个架构选项。以下重点介绍option2、3、4和7。 1. 独立组网 (Standalone, SA) 架构选项 2 &#xff1a;Standalone architecture with 5G-core 特点&#xff1a; 5G核心网&#xff08;5GC, …

数据分析实战—鸢尾花数据分类

1.实战内容 (1) 加载鸢尾花数据集(iris.txt)并存到iris_df中,使用seaborn.lmplot寻找class&#xff08;种类&#xff09;项中的异常值&#xff0c;其他异常值也同时处理 。 import pandas as pd from sklearn.datasets import load_iris pd.set_option(display.max_columns, N…

鸿蒙项目云捐助第十讲鸿蒙App应用分类页面二级联动功能实现

鸿蒙项目云捐助第十讲鸿蒙App应用分类页面二级联动功能实现 在之前的教程中完成了分类页面的左右两侧的列表结构&#xff0c;如下图所示。 接下来需要实现左侧分类导航项的点击操作&#xff0c;可以友好的提示用户选择了哪一个文字分类导航项。 一、左侧文字分类导航的处理 …

VSCode:Remote-SSH插件安装使用 -- 在VSCode中使用SSH

VSCode&#xff1a;Remote-SSH插件安装使用 1.安装Remote-SSH2.使用Remote-SSH 本文&#xff0c;将在Visual Studio Code中&#xff0c;安装Remote-SSH插件&#xff0c;实现SSH连接远程服务器。 1.安装Remote-SSH 打开VSCode&#xff0c;侧边栏中找到扩展模块(或CtrlShiftX快捷…

【机器人】Graspness 端到端 抓取点估计 | 论文解读

在复杂场景中实现抓取检测&#xff0c;Graspness是一种端到端的方法&#xff1b; 输入点云数据&#xff0c;输出抓取角度、抓取深度、夹具宽度等信息。 开源地址&#xff1a;GitHub - rhett-chen/graspness_implementation: My implementation of Graspnet Graspness. 论文地…

OpenWebUI,RAG+外部知识库+AI写文的开源应用

引言 自从去年AI火起来之后&#xff0c;很多人便热衷于寻找适合自用的AI开源项目&#xff0c;把各家大模型API接入到自己的AI程序里&#xff0c;便可以通过AI辅助完成一系列日常任务&#xff0c;比如内容翻译/润色/总结/撰写、格式转换、数据分类、代码分析、角色扮演等等。 …

力扣-图论-15【算法学习day.65】

前言 ###我做这类文章一个重要的目的还是给正在学习的大家提供方向和记录学习过程&#xff08;例如想要掌握基础用法&#xff0c;该刷哪些题&#xff1f;&#xff09;我的解析也不会做的非常详细&#xff0c;只会提供思路和一些关键点&#xff0c;力扣上的大佬们的题解质量是非…

单元测试知识总结

我们希望每段代码都是自测试的&#xff0c;每次改动之后&#xff0c;都能自动发现对现有功能的影响。 1 测试要求 在对软件单元进行动态测试之前&#xff0c;应对软件单元的源代码进行静态测试&#xff1b; 应建立测试软件单元的环境&#xff0c;如数据准备、桩模块、模拟器…

基于AI对话生成剧情AVG游戏

游戏开发这个领域&#xff0c;一直有较高的学习门槛。作为一个非专业的游戏爱好者&#xff0c;如果想要开发游戏&#xff0c;往往受制于游戏引擎的专业程度&#xff0c;难以完成复杂的游戏项目。 AI IDE的诞生&#xff0c;提供了另外的一种思路&#xff0c;即通过AI 生成项目及…