leetcode 41. 缺失的第一个正数(Java版)

题目描述(题目难度,困难)

给定一个未排序的整数数组,找出其中没有出现的最小的正整数。

示例 1:

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

示例 2:

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

示例 3:

输入: [7,8,9,11,12]
输出: 1

说明:
你的算法的时间复杂度应为O(n),并且只能使用常数级别的空间。

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/first-missing-positive
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

题目求解

这个题目的难点在于要求 O(n) 的时间复杂度,和 O(1) 的空间复杂度。

寻找缺失的第一个正数,需要先思考得出如下结论:
数组长度已知,假设为 n,则缺失的第一个正数最大只能是 n+1,其取值范围为 [1, n+1]。

因为对于一个长度为 n 的数组,缺失的第一个正数取最大值的情况是,数组中的 n 个数包括了从 1 到 n 的所有正数。此时缺失的第一个正数取最大值 n+1,不可能有比这还大的情况。

所以就算缺失的第一个正数取最大值,其前面的所有正数也都可以映射到数组下标上。

算法过程为,两趟遍历:

  • 第一趟遍历,判断数组元素是否在 1 到 数组长度 n 之间,如果在就把它放到数组的对应位置上。遍历结束后,值在 1 到 n 之间的元素,在数组中相对有序。
  • 第二趟遍历,寻找缺失的第一个正数。

参考代码如下:

class Solution {
    public int firstMissingPositive(int[] nums) {
        int i = 0, k, m = 0;
        while(i < nums.length){
        	if(nums[i] >= 1 && nums[i] <= nums.length && nums[k=nums[i]-1] != nums[i]){
				nums[k] ^= nums[i];
				nums[i] ^= nums[k];
				nums[k] ^= nums[i];
			}else ++i;
        }
        for(i = 0; i < nums.length; ++i){
            if(nums[i] - m == 1) m = nums[i];
        }
        return m+1;
    }
}

后记:数组是我们程序设计中最常用的数据结构,但即使是这种烂熟于心的东西,你能保证百分之百的掌握了吗?数组下标不仅是数组元素的索引,也是数组元素对应的隐含信息,没有任何的时空代价。所以在特定算法场景下,利用好数组下标,建立数组下标与数组值之间的联系,是使用数组的高级技巧。
大道至简,基础是通往大道的关键,各行各业拼到最后拼的都是基本功,所以在学习过程中,切勿好高骛远,打好坚实的基础最重要!

你可能感兴趣的:(算法,LeetCode)