其他
算法题320:寻找重复数
(给算法爱好者加星标,修炼编程内功)
转自:数据结构和算法/山大王wld
给定一个包含 n + 1 个整数的数组 nums,其数字都在 1 到 n 之间(包括 1 和 n),可知至少存在一个重复的整数。假设只有一个重复的整数,找出这个重复的数。
示例 1:
[1,3,4,2,2]
输出: 2
示例 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 -
觉得本文有帮助?请分享给更多人
关注「算法爱好者」加星标,修炼编程内功
好文章,我在看❤️