力扣448:找到所有数组中消失的数字(Java 鸽笼原理)

简介: 给你一个含 n 个整数的数组 nums ,其中 nums[i] 在区间 [1, n] 内。请你找出所有在 [1, n] 范围内但没有出现在 nums 中的数字,并以数组的形式返回结果。

一、题目描述



给你一个含 n 个整数的数组 nums ,其中 nums[i] 在区间 [1, n] 内。请你找出所有在 [1, n] 范围内但没有出现在 nums 中的数字,并以数组的形式返回结果。


示例 1:

输入:nums = [4,3,2,7,8,2,3,1]

输出:[5,6]


示例 2:

输入:nums = [1,1]

输出:[2]


提示:

n == nums.length

1 <= n <= 105

1 <= nums[i] <= n


进阶:你能在不使用额外空间且时间复杂度为 O(n) 的情况下解决这个问题吗? 你可以假定返回的数组不算在额外空间内。


二、思路讲解


     

显而易见,我们可以使用一个哈希表来记录数字是否出现。也可以用一个长度为n的数组flag代替哈希表,如果出现了数字i,就将flag[n-1]标记为1,最后看看有哪些位置为0就可以了。


class Solution {
    public List<Integer> findDisappearedNumbers(int[] nums) {
        int []flag = new int[nums.length];
        for(int num : nums) {
            flag[num-1] = 1;
        }
        List<Integer> list = new ArrayList<>();
        for(int i=0; i<flag.length; i++) {
            if(flag[i] == 0) {
                list.add(i+1);
            }
        }
        return list;
    }
}


时间复杂度:        O(N)


空间复杂度:        O(N)


但是,题目进阶要求我们不使用额外空间,那么我们可以看到,其实所给数字nums也是一个长度为n的数组,那么是不是可以考虑原地更改nums来做标记呢?


这就需要用到鸽笼原理,n个笼子,如果出现过,相应的“鸽笼”就会被占掉,我们可以将原数组中的值置为一个不可能出现的值,表示“鸽笼”已经被占掉。


那么问题就是如何能够得到该位置的原值。比较好的一种办法就是将原数字+n作为出现过的值,再将数字%n即可得到原值( 同样地,也可以通过对原数组的值取反来标记已经出现,通过取绝对值即可得到原值,在最后统计时,统计负数即可)。


需要注意的是,nums中的数字可以重复,那么某个位置有可能加过多个n,有溢出风险,需要对n取模防止溢出。

class Solution {
    public List<Integer> findDisappearedNumbers(int[] nums) {
        int n = nums.length;
        int []flag = new int[n];
        for(int i=0; i<n; i++) {
            int temp = (nums[i]-1) % n;
            nums[temp] = nums[temp] + n;
        }
        List<Integer> list = new ArrayList<>();
        for(int i=0; i<n; i++) {
            if(nums[i] <= n) {
                list.add(i+1);
            }
        }
        return list;
    }
}


时间复杂度:        O(N)

空间复杂度:        O(1)

目录
打赏
0
0
0
0
178
分享
相关文章
【原理】【Java并发】【synchronized】适合中学者体质的synchronized原理
本文深入解析了Java中`synchronized`关键字的底层原理,从代码块与方法修饰的区别到锁升级机制,内容详尽。通过`monitorenter`和`monitorexit`指令,阐述了`synchronized`实现原子性、有序性和可见性的原理。同时,详细分析了锁升级流程:无锁 → 偏向锁 → 轻量级锁 → 重量级锁,结合对象头`MarkWord`的变化,揭示JVM优化锁性能的策略。此外,还探讨了Monitor的内部结构及线程竞争锁的过程,并介绍了锁消除与锁粗化等优化手段。最后,结合实际案例,帮助读者全面理解`synchronized`在并发编程中的作用与细节。
36 8
【原理】【Java并发】【synchronized】适合中学者体质的synchronized原理
|
17天前
|
【原理】【Java并发】【volatile】适合初学者体质的volatile原理
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是写出高端的CRUD应用。2025年,我正在沉淀自己,博客更新速度也在加快。在这里,我会分享关于Java并发编程的深入理解,尤其是volatile关键字的底层原理。 本文将带你深入了解Java内存模型(JMM),解释volatile如何通过内存屏障和缓存一致性协议确保可见性和有序性,同时探讨其局限性及优化方案。欢迎订阅专栏《在2B工作中寻求并发是否搞错了什么》,一起探索并发编程的奥秘! 关注我,点赞、收藏、评论,跟上更新节奏,让我们共同进步!
88 8
【原理】【Java并发】【volatile】适合初学者体质的volatile原理
JVM实战—1.Java代码的运行原理
本文介绍了Java代码的运行机制、JVM类加载机制、JVM内存区域及其作用、垃圾回收机制,并汇总了一些常见问题。
JVM实战—1.Java代码的运行原理
|
13天前
|
《从头开始学java,一天一个知识点》之:数组入门:一维数组的定义与遍历
**你是否也经历过这些崩溃瞬间?** - 看了三天教程,连`i++`和`++i`的区别都说不清 - 面试时被追问&quot;`a==b`和`equals()`的区别&quot;,大脑突然空白 - 写出的代码总是莫名报NPE,却不知道问题出在哪个运算符 这个系列就是为你打造的Java「速效救心丸」!我们承诺:每天1分钟,地铁通勤、午休间隙即可完成学习;直击痛点,只讲高频考点和实际开发中的「坑位」;拒绝臃肿,没有冗长概念堆砌,每篇都有可运行的代码标本。明日预告:《多维数组与常见操作》。 通过实例讲解数组的核心认知、趣味场景应用、企业级开发规范及优化技巧,帮助你快速掌握Java数组的精髓。
57 23
Java快速入门之数组、方法
### Java快速入门之数组与方法简介 #### 一、数组 数组是一种容器,用于存储同种数据类型的多个值。定义数组时需指定数据类型,如`int[]`只能存储整数。数组的初始化分为静态和动态两种: - **静态初始化**:直接指定元素,系统自动计算长度,如`int[] arr = {1, 2, 3};` - **动态初始化**:手动指定长度,系统给定默认值,如`int[] arr = new int[3];` 数组访问通过索引完成,索引从0开始,最大索引为`数组.length - 1`。遍历数组常用`for`循环。常见操作包括求和、找最值、统计特定条件元素等。
Java 复制数组
本文介绍了Java中数组的基础知识与常用操作,包括数组的概念、创建、访问元素、遍历、复制、排序和搜索等方法。同时详细讲解了数组的五种赋值方式,并通过代码示例演示了求总和平均值、最大最小值、升序降序排序及Arrays类的常用方法。内容深入浅出,适合初学者学习掌握Java数组的核心功能与应用场景。
【JAVA】封装多线程原理
Java 中的多线程封装旨在简化使用、提高安全性和增强可维护性。通过抽象和隐藏底层细节,提供简洁接口。常见封装方式包括基于 Runnable 和 Callable 接口的任务封装,以及线程池的封装。Runnable 适用于无返回值任务,Callable 支持有返回值任务。线程池(如 ExecutorService)则用于管理和复用线程,减少性能开销。示例代码展示了如何实现这些封装,使多线程编程更加高效和安全。
Java基础(六):数组
Java基础(六):数组
36 10
Java基础(六):数组
【JAVA】生成accessToken原理
在Java中,生成accessToken用于身份验证和授权,确保合法用户访问受保护资源。流程包括:1. 身份验证(如用户名密码、OAuth 2.0);2. 生成唯一且安全的令牌;3. 设置令牌有效期并存储;4. 客户端传递令牌,服务器验证其有效性。常见场景为OAuth 2.0协议,涉及客户端注册、用户授权、获取授权码和换取accessToken。示例代码展示了使用Apache HttpClient库模拟OAuth 2.0获取accessToken的过程。
Java数组:静态初始化与动态初始化详解
本文介绍了Java中数组的定义、特点及初始化方式。
131 12