Leetcode287Find the Duplicate Number寻找重复数

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

你可能感兴趣的:(leetcode)