41. 缺失的第一个正数

leetcode第41题,缺失的第一个正数

对于这道题,利用的是桶排序的思想,桶排序也就是利用数组的下标和值对应的关系,也就是说下标为0的索引对应的值应该就是0,依次类推n对应n

但是这只是一种对应关系,也就是说我们不仅仅只局限于这种关系,只要有合理的对应关系即可,那么这题要求从1开始第一个没有的正整数。所以我们将下标0对应1

n-1对应n。

考虑正整数最小为1,也就是不能缺少1否则答案直接就是1,而且数组长度为n,那么也就是说一共才有n个数,又要求从1开始,所以结果应该在[1,n+1]这个区间之间

我们是想一下极端情况,也就是说数组就是连续的且从1开始,那么我们从最小开始遍历得到的结果就是最后的n+1,

所以进一步考虑在排序的时候每一个空间不能浪费,也就是0是最小的,那么就对应1。

遍历数组,如果这个数小于等于0直接跳过,如果大于n也跳过。

如果这个数大于0也小于等于n,但是这个数不在它应该在的位置,也就是比如说,在索引3有一个1,那么这个1应该在0这个索引的位置,所以我们需要将0索引的值和这个3索引的值交换一下,同时我们交换来的这个值,我们还没有判断这个值的情况,也就是说不能直接跳过它去下一个位置,此时我们需要i--,(这个地方的处理就是在快排的时候对应大于v的时候交换位置时候的处理)

在交换完成之后,也就是数组遍历完成,我们再次遍历数组寻找那个缺失的正整数,如果发现下标为i的地方的值不为i+1,那么结果就是i+1是缺失的了,直接返回这个值即可,如果遍历到尾还没有找到,说明这个数组排好序以后是一个有序的连续的数组,那么我们在这个循环的外面直接返回一个n+1即可。

下面是代码:

 1 class Solution {
 2     public int firstMissingPositive(int[] nums) {
 3         int n=nums.length;
 4        for(int i=0;i<n;i++)
 5        {
 6            if(nums[i]>0&&nums[i]<=n&&nums[nums[i]-1]!=nums[i])
 7            {
 8                 swap(nums,i,nums[i]-1);
 9                i--;
10            }
11        }
12         for(int i=0;i<n;i++)
13         {
14             if(nums[i]!=i+1)
15                 return i+1;
16         }
17         return n+1;
18     }
19     public void swap(int[] arr,int l,int r){
20         int temp=arr[l];
21         arr[l]=arr[r];
22         arr[r]=temp;
23     }
24 }

 

posted @ 2019-10-23 09:04  小路不会迷路  阅读(150)  评论(0编辑  收藏  举报