Leetcode287寻找重复数
题目描述:
在一个长度为n+1的数组中,每个数都是1-n之间,只有一个数出现两次,其他的数都只出现过一次,请找出这个数。
输入:
长度为n的数组
输出:
重复的数字num
要求:
时间复杂度O(N),空间复杂度O(1)
思路:
1.使用set集合的方式,但是很明显,空间复杂度是O(N),不符合。
2.先对数组进行排序,然后逐位进行比较,很明显,时间复杂度是O(NlogN),空间复杂度是O(1),不符合。
3.观察题目,1-n,刚好n+1的数组,所以将数字放入对应的下标中,在放入之间判断是否两者是否已经相同,如果相同就返回,如果不同就交换位置直到有相同的为止。
public int getSameNum(int[] nums) {
if (nums == null || nums.length == 0) {
return -1;
}
for (int i = 0; i < nums.length; i++) {
while (i != nums[i]) {
if (nums[i] == nums[nums[i]]) {
System.out.println(nums[i]);
return nums[i];
}
int tmp = nums[nums[i]];
nums[nums[i]] = nums[i];
nums[i] = tmp;
}
}
return -1;
}
解释:
外面的for循环是为了保证每个都能遍历到,当下标为i的位置存放的不是i时就进入while循环进行调换,在while循环中,如果下标为i的位置的数字和以该数字为下标的位置的数字相同(就是即将要调换位置的两个数),则已经找出这两个数,如果不同就进行调换,将nums[nums[i]]的数字放在其下标nums[i]对应的位置中,然后将没有对应入座的num[num[i]]赋值给nums[i],从而进行while循环。如果所有的数字都遍历完(for循环结束)则返回-1.
怎么保证while循环不会死循环:
- 第一,如果有两个相同的数,这样在循环的过程中将每个数放入对应的下标中(即3放入数组下标为3的地方,即nums[3] = 3),因为是在调换之前比较的,所以一定可以遍历完到两个相同的值。
- 第二,如果不存在相同的数,当所有的数都对应下标入座后,for循环会结束。
zsjwish
2018年4月25日21:51:59