锑砖 发表于 2025-12-16 10:10:00

剑指offer-50、数组中重复的数字

题目描述

在⼀个⻓度为 n 的数组⾥的所有数字都在 0 到n-1 的范围内。 数组中某些数字是重复的,但不知
道有⼏个数字是重复的。也不知道每个数字重复⼏次。请找出数组中第⼀个重复的数字。 例如,如果输⼊⻓度为 7 的数组 ,那么对应的输出是第⼀个重复的数字 2 。没有重复的数字
返回 -1 。
示例1
输⼊
[ 2, 3, 1, 0, 2, 5, 3 ]
返回值
2
思路及解答

借用Set

⾸先可能想到的做法,就是借助 set ,如果元素不存在 set 中,就将元素添加进去,如果 set
中包含该元素,就返回该元素即可。如果⼀直都没有重复的,那么最后返回 -1 。
public class Solution {
        public int duplicate(int[] numbers) {
                if (numbers != null) {
                        Set<Integer> set = new HashSet<>();
                        for (int i = 0; i < numbers.length; i++) {
                                if (set.contains(numbers)) {
                                        return numbers;
                                } else {
                                        set.add(numbers);
                                }
                        }
                }
                return -1;
        }
}

[*]时间复杂度:O(n) ,最差的情况可能遍历完所有的元素
[*]空间复杂度: O(n) ,最⼤需 要set ⼤⼩为 n
借助数组

可以直接借助数组,因为所有数字都在 0 到 n-1 的范围内,⽤⼀个⼤⼩为 n 的数组,就可以对所有的数字进⾏统计个数,如果个数超过 1 ,那么肯定是重复的数字,如果没有重复的数字,则返回 -1 ;
public class Solution {
        public int duplicate(int[] numbers) {
                if (numbers != null) {
                        int[] nums = new int;
                        for (int i = 0; i < numbers.length; i++) {
                                if (nums] == 1) {
                                        return numbers;
                                } else {
                                        nums] = 1;
                                }
                        }
                }
                return -1;
        }
}同样这种做法的时间复杂度和空间复杂度都是 O(n) ,并没有优化太多。
那么有没有空间复杂度为O(1) 的做法呢?
操作原数组(最优)

不借助额外的空间,那么就只能操作原数组了。如果没有重复的情况,那么这些数字排序后,数字i 和数组下标i 应该是⼀⼀对应的。不会出现多个数字i 的情况。
基于这个原则,在遍历数组的时候,将元素 i 调整到下标 i 的位置,如果下标i的位置已经有元
素,那么说明冲突了,这个元素肯定是重复的,否则继续调整后⾯的。如果没有发现重复的数字,就返回 -1 。
public class Solution {
        public int duplicate(int[] numbers) {
                int i = 0;
                while(i < numbers.length) {
                        if(numbers == i) {
                                i++;
                                continue;
                        }
                        if(numbers] == numbers) return numbers;
                        int tmp = numbers;
                        numbers = numbers;
                        numbers = tmp;
                }
                return -1;
        }
}但是上⾯的做法,不适合求解多个重复数字的例⼦,因为调换的时候,很容易将后⾯的数字换到前⾯去,就会导致求解出来不是第⼀个重复的数字(可以⽤来求解任意的重复数字),可能是第2,3... 或者其他的重复数字。譬如: 正确的解应该是 2 ,但是由于第⼀次把 6 和最后
的0 调换了位置,就会导致求解出来的值为 0 。

来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

宁觅波 发表于 2025-12-21 12:42:55

感谢分享,下载保存了,貌似很强大

赖秀竹 发表于 2025-12-22 17:59:48

谢谢分享,辛苦了

蓝娅萍 发表于 2025-12-24 22:38:48

谢谢分享,辛苦了

丝甲坞 发表于 2026-1-7 20:54:28

分享、互助 让互联网精神温暖你我

菅舛 发表于 2026-1-9 01:46:01

收藏一下   不知道什么时候能用到

孜稞 发表于 2026-1-15 05:01:41

不错,里面软件多更新就更好了

辉伫 发表于 2026-1-15 22:50:28

感谢分享,下载保存了,貌似很强大

姜删懔 发表于 2026-1-19 22:24:42

热心回复!

赐度虻 发表于 2026-1-20 08:22:52

热心回复!

后沛若 发表于 2026-1-23 09:13:34

yyds。多谢分享

祝安芙 发表于 2026-1-25 03:31:07

新版吗?好像是停更了吧。

诀锺 发表于 2026-1-27 02:41:45

东西不错很实用谢谢分享

站竣凰 发表于 2026-1-28 05:48:32

鼓励转贴优秀软件安全工具和文档!

指陡 发表于 2026-1-29 02:25:30

懂技术并乐意极积无私分享的人越来越少。珍惜

髡芯 发表于 2026-2-6 04:48:51

热心回复!

裒噎 发表于 2026-2-7 12:04:47

谢谢楼主提供!

致掣 发表于 2026-2-8 12:39:31

感谢分享,学习下。

祝娜娜 发表于 2026-2-8 15:00:04

感谢,下载保存了

裴竹悦 发表于 2026-2-9 07:46:17

懂技术并乐意极积无私分享的人越来越少。珍惜
页: [1] 2
查看完整版本: 剑指offer-50、数组中重复的数字