查看原文
其他

算法题320:寻找重复数

(给算法爱好者加星标,修炼编程内功

转自:数据结构和算法/山大王wld

给定一个包含 n + 1 个整数的数组 nums,其数字都在 1 到 n 之间(包括 1 和 n),可知至少存在一个重复的整数。假设只有一个重复的整数,找出这个重复的数。


示例 1:


输入:[1,3,4,2,2]
输出: 2


示例 2:


输入: [3,1,3,4,2]
输出: 3


答案:


1public int findDuplicate(int[] nums) {
2    HashMap hashMap = new HashMap();
3    for (int i = 0; i < nums.length; i++) {
4        if (hashMap.put("" + nums[i], "" + nums[i]) != null)
5            return nums[i];
6    }
7    return -1;
8}


解析:


这题基本上没什么难度,解法也比较多,其实我们还可以排序,然后再查找,下面再来看一种解法



1public int findDuplicate(int[] nums) {
2    int n = nums.length - 1;
3    int low = 1;
4    int high = n;
5    int mid;
6    while (low < high) {
7        mid = (low + high) / 2;
8        int count = 0;
9        for (int num : nums) {
10            if (num <= mid)
11                count++;
12        }
13        if (count > mid)
14            high = mid;
15        else
16            low = mid + 1;
17    }
18    return low;
19}


二分法查找我们一般是用在已经排序过的数组,但这里很明显是没有排过序的,但也不影响。因为在while循环中有个for循环来进行计算,如果count大于mid(中间值),说明重复的数字肯定是在数组的前面,因为数字是从1到n然后再包含一个重复的数字。其实这题还有一种非常巧妙的解法,我们可以参考275,环形链表 II这题的思路,看下代码


1public int findDuplicate(int[] nums) {
2    if (nums.length > 1) {
3        int slow = nums[0];
4        int fast = nums[nums[0]];
5        while (slow != fast) {
6            slow = nums[slow];
7            fast = nums[nums[fast]];
8        }
9        fast = 0;
10        while (fast != slow) {
11            fast = nums[fast];
12            slow = nums[slow];
13        }
14        return slow;
15    }
16    return -1;
17}


因为元素的大小是从1到n,有一个是重复的,所以我们可以通过下标和值映射成一个环,下面看个图来帮助理解,我们以【1,3,4,2,3】来举例说明



其实我们还可以把数组的值全部加一遍,然后通过公式计算1到n的值,最后进行相减即可,这里就不再列出。


- EOF -


推荐阅读  点击标题可跳转

1、图解排序算法:快速排序

2、从 LRU Cache 带你看面试的本质

3、机器学习算法优缺点综述


觉得本文有帮助?请分享给更多人

关注「算法爱好者」加星标,修炼编程内功

好文章,我在看❤️

    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存