题目:
在一个长度为 n 的数组里的所有数字都在 0 到 n-1 的范围内。 数组中某些数字是重复的,但不知道有几个数字是重复的,也不知道每个数字重复几次。 请找出数组中任意一个重复的数字。分析
- 时间复杂度O(n),空间复杂度O(1)。这样肯定不能新建数组和排序遍历
一个长度为 n 的数组里的所有数字都在 0 到 n-1 的范围内
,关键信息点
数组下标为0到n-1不重复的数字
如果不重复,把每个元素都可以放在对应下标位置
思路
遍历每个元素,如果元素和对应下标不相等,需要把元素放在元素值的下标位置。元素放入对应位置时,存在值且相等,则重复。
源码
package org.example.array;
/**
* 数组中重复的数字
* @author: guochao.bj@fang.com
* @createDate: 2022/3/16 19:38
*/
public class practiceArray01 {
/**
* 题目
* 在一个长度为 n 的数组里的所有数字都在 0 到 n-1 的范围内。
* 数组中某些数字是重复的,但不知道有几个数字是重复的,也不知道每个数字重复几次。
* 请找出数组中任意一个重复的数字。
* https://github.com/CyC2018/CS-Notes/blob/master/notes/3.%20%E6%95%B0%E7%BB%84%E4%B8%AD%E9%87%8D%E5%A4%8D%E7%9A%84%E6%95%B0%E5%AD%97.md
* Input:
* {2, 3, 1, 0, 2, 5}
* Output:
* 2
*/
public static void main(String[] args) {
int[] test={2, 3, 1, 0, 2, 5};
int duplicate = duplicate(test);
System.out.println(duplicate);
}
/**
* 解题思路
* 1.时间复杂度O(n),空间复杂度O(1)。这样肯定不能新建数组和排序遍历
* 2.一个长度为 n 的数组里的所有数字都在 0 到 n-1 的范围内,关键信息点
* 说明:a>:数组下标为0到n-1不重复的数字
* b>:如果不重复,把每个元素都可以放在对应下标位置
*/
public static int duplicate(int[] nums) {
for (int i = 0; i < nums.length; i++) {
while (nums[i] != i) {
//如果对应位置已经存在值,则说明重复
if (nums[i] == nums[nums[i]]) {
return nums[i];
}
swap(nums, i, nums[i]);
}
}
return -1;
}
/**
* 把数据放到对应位置上去
*/
private static void swap(int[] nums, int i, int j) {
int t = nums[i];
nums[i] = nums[j];
nums[j] = t;
}
}