Leetcode尊享面试100题:1060. 有序数组中的缺失元素

现有一个按 升序 排列的整数数组 nums ,其中每个数字都 互不相同

给你一个整数 k ,请你找出并返回从数组最左边开始的第 k 个缺失数字。

示例 1:

复制代码
输入:nums = [4,7,9,10], k = 1
输出:5
解释:第一个缺失数字为 5 。

示例 2:

复制代码
输入:nums = [4,7,9,10], k = 3
输出:8
解释:缺失数字有 [5,6,8,...],因此第三个缺失数字为 8 。

示例 3:

复制代码
输入:nums = [1,2,4], k = 3
输出:6
解释:缺失数字有 [3,5,6,7,...],因此第三个缺失数字为 6 。

提示:

  • 1 <= nums.length <= 5 * 104
  • 1 <= nums[i] <= 107
  • nums升序 排列,其中所有元素 互不相同
  • 1 <= k <= 108

进阶: 你可以设计一个对数时间复杂度(即,O(log(n)))的解决方案吗?

直接上代码,不会的可以私信或者留言:

java 复制代码
class Solution {
    //基础的解法是:数一下每两个数之间缺几个数,如果从头开始数到这个区间仍然小于k,就用k
    //剪去这个区间缺的数量,如果大于等于k了就停止
    public int missingElement2(int[] nums, int k) {
        int remains = k;
        //每个区间都不缺就是缺在最后一个数的后面
        int stopIndex = nums.length - 1;
        for(int i = 1;i < nums.length; i++) {
            //i-1~i这个区间缺了多少个数
            int missingInCurRange = nums[i] - nums[i - 1] - 1;
            //如果这个区间缺的数还是不够就继续搜索下个区间
            if(missingInCurRange < remains) {
                remains -= missingInCurRange;
            } else {
                //否则缺的数就在本区间,本区间的起点是i-1
                stopIndex = i - 1;
                break;
            }
        }
        return nums[stopIndex] + remains;
    }

    //进阶的解法,如果要实现logn的解法,必定是使用二分查找,我也想不到什么特别好的方案
    //只能说我试试,比如找到数组中某个位置,如果到以这个位置为起点的区间确实的数小于等于k就继续尝试找更大的
    //符合缺失数小于等于k的,如果大于k就往回找
    public int missingElement(int[] nums, int k) {
        //如果就一个数,缺k个就用第一个数+k返回
        if(nums.length == 1) {
            return nums[0] + k;
        }
        //使用二分法查找:区间结尾怎么也得是下标为1的位置了,最大是nums.length - 1
        int l = 1;
        int r = nums.length - 1;
        //没有证明前面的区间缺这么多的时候只能认为是在最后一个数的后面缺
        int endIndex = nums.length;
        //二分查找
        while(l <= r) {
            int mid = l + (r - l)/2;
            //到mid这个位置结尾的区间缺了多少个数
            int missingCount = missingCount(nums, mid);
            //如果到mid结尾的位置已经超过k个了,说明就在这个区间或者在上一个区间
            if(missingCount >= k) {
                //先记录一个答案,然后继续往前找
                endIndex = mid;
                r = mid - 1;
            } else {
                //不够k说明还得继续往后
                l = mid + 1;
            }
        }
        /**在0~endIndex-1这个区间缺了多少个数,如果不缺的话应该是nums[endIndex - 1] - nums[0] + 1个数
        但是现在只有endIndex个数,所以缺了(nums[endIndex - 1] - nums[0] + 1) - endIndex个*/
        int preMissing = ((nums[endIndex - 1] - nums[0] + 1) - endIndex);
        return nums[endIndex - 1] + (k - preMissing);
    }
    //到以index这个位置为终点的区间一共缺失了多少个数
    public int missingCount(int[] nums, int index) {
        return nums[index] - nums[0] + 1 - (index + 1);
    }
}

运行效果:

相关推荐
j7~10 小时前
【算法】专题一:双指针之移动零,复写零,快乐数
数据结构·c++·算法·双指针·快乐数·移动零·复写零
lqqjuly10 小时前
KAN 网络深度解析
算法
阿里matlab建模师10 小时前
【机场停机位分配】matlab实现基于遗传算法的机场停机位分配优化研究
开发语言·算法·数学建模·matlab·全国大学生数学建模竞赛
小雨下雨的雨16 小时前
井字棋AI机器人实现详解 - Minimax算法实战-鸿蒙PC Electron框架完成
前端·人工智能·算法·华为·electron·鸿蒙
xieliyu.18 小时前
Java算法精讲:双指针(三)
java·开发语言·算法
一条小锦吕*18 小时前
基于Spring Boot + 数据可视化 + 协同过滤算法的推荐系统设计与实现(源码+论文+部署全讲解)
spring boot·算法·信息可视化
綝~19 小时前
爬虫数据采集工程师岗位面试题
爬虫·面试·请求
如竟没有火炬20 小时前
最大矩阵——单调栈
数据结构·python·线性代数·算法·leetcode·矩阵
8Qi821 小时前
LeetCode 1143 & 718:最长公共子序列 / 最长重复子数组
算法·leetcode·职场和发展·动态规划
绿算技术21 小时前
万卡推理集群存储选型分析:从核心架构到应用视角
大数据·科技·算法·架构