力扣1539. 第 k 个缺失的正整数

这一题的大意是说给出一个数组,和一个整数k,这个数组是严格升序排列的,现在让我们找第k个缺失的正整数。

什么叫做第k个缺失的正整数呢?意思是数组应该是从自然数1开始1...2...3正常排列的,但现在有缺失的,让我们找第k个缺失的数是多少?

那么我们如何可以判断呢,题目上让用一个小于logn的算法,又因为是让求第K个缺失的数,很明显是用二分。

那么该怎么二分呢?

我们很容易发现要想判断某一数组中缺失多少个数,只需要用 arr[i]-(i+1) (i从0开始)即可求得。

那么我们只需要找第一个大于等于k的数,那么在它的上一个数就是最后一个小于等于k的数,我们只需要用arr[l-1]+k-arr[l-1]-l即算出第k个缺失的数是多少。这里的l表示的是第一个大于等于k的数,而l-1表示的是最后一个小于等于k的数。

这里的第一个arr[l-1]表示从这个往后再加+k-arr[l-1]-l个数就是第k个缺失的数。

而k-arr[l-1]-l表示最后一个小于等于k的数之前已经缺失了这些数,但还没有达到k个,还差k-arr[l-1]-l个数就达到了缺失的第k个数,我们只需要用arr[l-1]+k-arr[l-1]-l即为第k个缺失的数,化简得 k+l

因此代码如下:

cpp 复制代码
class Solution {
public:
    int findKthPositive(vector<int>& arr, int k) {
        int l=0;
        int r=arr.size()-1;
        while(l<=r)
        {
            int mid=(l+r)/2;
            if(arr[mid]-mid-1>=k)
            {
                r=mid-1;
            }
            else
            {
                l=mid+1;
            }
        }
        if(l!=0)
         return (arr[l-1]+(k-(arr[l-1]-l)));
        else{
            return k;
        }
    }
};

或者

cpp 复制代码
class Solution {
public:
   
    int findKthPositive(vector<int>& arr, int k) {
        int l=0;
        int r=arr.size()-1;
       
        while(l<=r)
        {
            int mid=(l+r)/2;
            if(arr[mid]-mid-1>=k)
            {
                r=mid-1;
            }
            else
            {
                l=mid+1;
            }
        }
         return l+k;
        
    }
};

这一题很容易想到二分,想到找第一个大于等于k的数的位置,但对于第k个数是多少,我觉得很不好想,看题解都是return l+k,但说不清楚l+k的含义。

实际上l+k的含义就是arr[l-1]+k-arr[l-1]-l的数学化简,要理解还是要用arr[l-1]+k-arr[l-1]-l

arr[l-1]+k-arr[l-1]-l即算出第k个缺失的数是多少。

时间复杂度O(logn)

我们也可以直接找最后一个小于k的数的位置,然后直接用arr[r]+k-(arr[r]-(r+1))好理解。

cpp 复制代码
class Solution {
public:
    int findKthPositive(vector<int>& arr, int k) {
        int l=0;
        int r=arr.size()-1;
        while(l<=r)
        {
            int mid=(l+r)/2;
            if(arr[mid]-mid-1<k)
            {
                l=mid+1;
            }
            else
            {
                r=mid-1;
            }
        }
       //最后一个小于k的位置
      if(r==-1)
      {
        return k;
      }
      else
       return arr[r]+(k-(arr[r]-(r+1)));
    
    }
};

这里的r是最终指向的最后一个小于k的位置,这里是跟上面的l的意思是类似的。

我们直接用二分找到最后一个缺失数字小于k的位置用这个位置的值arr[r]+(k-(arr[r]-r-1)(还差多少个数字)从而表示出第k个缺失的数字。

为什么非要找最后一个小于k的位置或者第一个大于等于k的位置,关键就在于我们可以弄清楚 第k个数就在最后一个小于k的位置或者第一个大于等于k的位置之间,只需要在再原有的基础上加上还缺的数字个数即可。

相关推荐
逻辑驱动的ken1 分钟前
Java高频面试考点场景题09
java·开发语言·数据库·算法·oracle·哈希算法·散列表
帅小伙―苏16 分钟前
力扣42接雨水
前端·算法·leetcode
AI科技星32 分钟前
精细结构常数α的几何本源:从第一性原理的求导证明、量纲分析与全域验证
算法·机器学习·数学建模·数据挖掘·量子计算
6Hzlia37 分钟前
【Hot 100 刷题计划】 LeetCode 287. 寻找重复数 | C++ 数组判环 (快慢指针终极解法)
c++·算法·leetcode
MegaDataFlowers38 分钟前
26.删除有序数组中的重复项
算法
故事和你912 小时前
洛谷-数据结构1-4-图的基本应用2
开发语言·数据结构·算法·深度优先·动态规划·图论
吴可可1232 小时前
C#合并首尾相连多段线实战
算法·c#
KMDxiaozuanfeng3 小时前
卡梅德生物技术快报|SPR 技术应用|基于 SPR 亲和力的中药活性成分筛选系统实现与数据分析
科技·算法·面试·考试
꧁细听勿语情꧂3 小时前
数据结构概念和算法、时间复杂度、空间复杂度引入
c语言·开发语言·数据结构·算法
Felven4 小时前
B. The 67th 6-7 Integer Problem
数据结构·算法