ShuangChenYue ShuangChenYue
首页
  • Cpp之旅
  • Cpp专栏
  • Effective_CPP
  • muduo网络库
  • Unix环境高级编程
  • Cpp提高编程
  • 计算机网络
  • 操作系统
  • 数据结构
  • Linux
  • 算法
  • 基础篇
  • MySql
  • Redis
  • 电子嵌入式通信协议
  • 深入浅出SSD
  • 文件系统
  • 汇编语言
  • STM32
  • 随笔(持续更新)
  • Git知识总结
  • Git 创建删除远程分支
  • nvm使用小结
  • 虚拟机固定 IP 地址
  • Shell 脚本学习笔记
  • VScode 插件 CodeGeeX 使用教程
  • KylinV10 将项目上传至 Github教程
  • KylinV10 安装 MySQL 教程(可防踩雷)
  • kylinV10-SP1 安装 QT
  • 高并发内存池
  • USBGUARD 项目编译环境配置
  • Power_Destory 项目
  • U 盘清除工具编译教程
  • 个人博客代码推送教程
  • HTML与CSS
  • JS学习
  • Vue3入门
  • Vue3进阶
  • 黑马Vue3
  • MFC编程随记
  • MFC实现ini配置文件的读取
  • MFC实现点击列表头排序
  • 贴图法美化Button按钮
  • 如何高效阅读嵌入式项目代码
  • NAND Flash
  • ARM 处理器
  • 嵌入式基础知识-存储器
  • 闪存存储和制造技术概述
  • 芯片IO驱动力
  • 主流先进封装技术介绍
  • 虎牙C++技术面经
  • 金山一面复习
  • 完美世界秋招 C++ 游戏开发面经(Cpp部分)
  • 博客搭建
  • 网站收藏箱
首页
  • Cpp之旅
  • Cpp专栏
  • Effective_CPP
  • muduo网络库
  • Unix环境高级编程
  • Cpp提高编程
  • 计算机网络
  • 操作系统
  • 数据结构
  • Linux
  • 算法
  • 基础篇
  • MySql
  • Redis
  • 电子嵌入式通信协议
  • 深入浅出SSD
  • 文件系统
  • 汇编语言
  • STM32
  • 随笔(持续更新)
  • Git知识总结
  • Git 创建删除远程分支
  • nvm使用小结
  • 虚拟机固定 IP 地址
  • Shell 脚本学习笔记
  • VScode 插件 CodeGeeX 使用教程
  • KylinV10 将项目上传至 Github教程
  • KylinV10 安装 MySQL 教程(可防踩雷)
  • kylinV10-SP1 安装 QT
  • 高并发内存池
  • USBGUARD 项目编译环境配置
  • Power_Destory 项目
  • U 盘清除工具编译教程
  • 个人博客代码推送教程
  • HTML与CSS
  • JS学习
  • Vue3入门
  • Vue3进阶
  • 黑马Vue3
  • MFC编程随记
  • MFC实现ini配置文件的读取
  • MFC实现点击列表头排序
  • 贴图法美化Button按钮
  • 如何高效阅读嵌入式项目代码
  • NAND Flash
  • ARM 处理器
  • 嵌入式基础知识-存储器
  • 闪存存储和制造技术概述
  • 芯片IO驱动力
  • 主流先进封装技术介绍
  • 虎牙C++技术面经
  • 金山一面复习
  • 完美世界秋招 C++ 游戏开发面经(Cpp部分)
  • 博客搭建
  • 网站收藏箱
  • 网络

  • 操作系统

  • 数据结构

  • 算法

    • 两数之和
    • 回文数
    • 最长公共前缀
    • 三数之和
    • 删除有序数组中的重复项
    • 最大子数组和
    • x 的平方根
    • 爬楼梯
    • 对称二叉树
    • 二叉树的最大深度
    • LCR寻找文件副本
    • 买卖股票的最佳时机
    • LCR图书整理 II
    • 只出现一次的数字
      • 题目:
      • 示例:
      • 解题:
        • 方法一:集合法
        • 方法二:哈希表法
        • 方法三:元素之和两倍性质
        • 方法四:位运算(线性时间复杂度,常数空间复杂度)
    • LCR 训练计划 II
    • 环形链表
    • LRU 缓存
    • 反转字符串中的单词
    • LCR 破冰游戏
    • 反转链表
    • 翻转二叉树
    • 回文链表
    • 移动零
    • 最长回文串
    • 汉明距离
    • 把二叉搜索树转换为累加树
    • 最短无序连续子数组
    • 合并二叉树
    • 二分查找
    • 链表的中间结点
    • 有序数组的平方
    • 找到小镇的法官
  • Linux

  • 计算机基础
  • 算法
霜晨月
2023-11-24
目录

只出现一次的数字

# 136. 只出现一次的数字 (opens new window)

# 题目:

给你一个 非空 整数数组 nums ,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素。

你必须设计并实现线性时间复杂度的算法来解决此问题,且该算法只使用常量额外空间。

# 示例:

示例 1 :

输入:nums = [2,2,1]
输出:1
1
2

示例 2 :

输入:nums = [4,1,2,1,2]
输出:4
1
2

示例 3 :

输入:nums = [1]
输出:1
1
2

提示:

  • 1 <= nums.length <= 3 * 104
  • -3 * 104 <= nums[i] <= 3 * 104
  • 除了某个元素只出现一次以外,其余每个元素均出现两次。

# 解题:

如果不考虑时间复杂度和空间复杂度的限制方法有很多:

# 方法一:集合法

使用集合unordered_set存储数字。遍历数组中的每个数字,如果集合中没有该数字,则将该数字加入集合,如果集合中已经有该数字,则将该数字从集合中删除,最后剩下的数字就是只出现一次的数字。

class Solution {
public:
    int singleNumber(vector<int>& nums) {
        unordered_set<int> numSet;
        for(int num : nums) {
            // 如果集合中已经有当前数字,则从集合中删除
            if(numSet.find(num) != numSet.end()) {
                numSet.erase(num);
            } else {
                // 如果集合中没有当前数字,则加入集合
                numSet.insert(num);
            }
        }
        // 集合中剩下的就是只出现一次的数字
        return *numSet.begin();
    }
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17

# 方法二:哈希表法

使用哈希表存储每个数字和该数字出现的次数。遍历数组即可得到每个数字出现的次数,并更新哈希表,最后遍历哈希表,得到只出现一次的数字。

class Solution {
public:
    int singleNumber(vector<int>& nums) {
        unordered_map<int,int> numCount;
        // 遍历数组,更新哈希表中数字的出现次数
        for(int num : nums) {
            numCount[num]++;
        }
        // 遍历哈希表,找到只出现一次的数字
        for(auto& pair : numCount) {
            if(pair.second == 1) {
                return pair.first;
            }
        }
        // 如果没有找到只出现一次的数字,返回默认值0
        return 0;
    }
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

# 方法三:元素之和两倍性质

由于集合保证元素无重复,所以使用集合unordered_set不重复的存储数组的元素,也就是每个元素只存储一次,重复的不存储,计算它们的和,就相当于所有数字的两倍之和。然后将原数组中的元素全部相加,就相当于只出现了一次的元素加上全部出现了两次的元素。如此看来,它们的差就是就差了一个只出现一次的元素了。

class Solution {
public:
    int singleNUmber(vector<int>& nums) {
        unordered_set<int> numSet;
        int sumSet = 0;
        int sumArray = 0;
        // 遍历数组,更新集合中的元素之和和数组中的元素之和
        for(int num : nums) {
            if(numSet.find(num) == numSet.end()) {
                numSet.insert(num);
                sumSet += num;
            }
            sumArray += num;
        }
        // 计算集合中的元素之和的两倍减去数组中的元素之和,得到只出现一次的数字
        return 2*sumSet - sumArray;
    }
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

上述三种解法都需要额外使用 O(n) 的空间,其中 n 是数组长度。

如何才能做到线性时间复杂度和常数空间复杂度呢?

# 方法四:位运算(线性时间复杂度,常数空间复杂度)

异或运算有以下三个性质:

  1. 任何数和 0 做异或运算,结果仍然是原来的数,即 a⊕0=a。

  2. 任何数和其自身做异或运算,结果是 0,即 a⊕a=0。

  3. 异或运算满足交换律和结合律,即 a⊕b⊕a=b⊕a⊕a=b⊕(a⊕a)=b⊕0=b。

1700732063880

1700732089334

1700732110833

假设数组中有 2m+1 个数,其中有 m 个数各出现两次,一个数出现一次。

令 a~1~ 、a~2~ 、…、a~m~为出现两次的 m 个数,a~m+1~为出现一次的数。

根据性质 3,数组中的全部元素的异或运算结果总是可以写成如下形式:

  • (a~1~⊕a~1~)⊕(a~2~⊕a~2~)⊕⋯⊕(a~m~⊕a~m~)⊕a~m+1~

根据性质 2 和性质 1,上式可化简和计算得到如下结果:

  • 0⊕0⊕⋯⊕0⊕a~m+1~=a~m+1~

因此,数组中的全部元素的异或运算结果即为数组中只出现一次的数字。

class Solution {
public:
    int singleNumber(vector<int>& nums) {
        int ret = 0;
        for(auto e : nums) ret ^= e;
        return ret;
    }
};
1
2
3
4
5
6
7
8

复杂度分析

  • 时间复杂度:O(n),其中 n 是数组长度。只需要对数组遍历一次。
  • 空间复杂度:O(1)。
上次更新: 2024/6/3 14:54:44
LCR图书整理 II
LCR 训练计划 II

← LCR图书整理 II LCR 训练计划 II→

Theme by Vdoing | Copyright © 2023-2024 霜晨月
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式