美文网首页
leetcode 41. 缺失的第一个正数(Java版)

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

作者: M_lear | 来源:发表于2019-06-19 15:16 被阅读0次

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

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

    示例 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 41. 缺失的第一个正数(Java版)

          本文链接:https://www.haomeiwen.com/subject/dmxkqctx.html