mirror of
https://github.com/krahets/LeetCode-Book.git
synced 2026-01-12 00:19:02 +08:00
2.7 KiB
Executable File
2.7 KiB
Executable File
解题思路:
排序数组中的搜索问题,首先想到 二分法 解决。根据题意,数组可以按照以下规则划分为两部分。
- 左子数组:
records[i] = i; - 右子数组:
records[i] \ne i;
缺失的数字等于 “右子数组的首位元素” 对应的索引;因此考虑使用二分法查找 “右子数组的首位元素” 。
下图中的
nums对应本题的records。
算法解析:
- 初始化: 左边界
i = 0,右边界j = len(records) - 1;代表闭区间[i, j]。 - 循环二分: 当
i \leq j时循环 (即当闭区间[i, j]为空时跳出) ;- 计算中点
m = (i + j) // 2,其中 "$//$" 为向下取整除法; - 若
records[m] = m,则 “右子数组的首位元素” 一定在闭区间[m + 1, j]中,因此执行 $i = m + 1$; - 若
records[m] \ne m,则 “左子数组的末位元素” 一定在闭区间[i, m - 1]中,因此执行 $j = m - 1$;
- 计算中点
- 返回值: 跳出时,变量
i和j分别指向 “右子数组的首位元素” 和 “左子数组的末位元素” 。因此返回i即可。
代码:
class Solution:
def takeAttendance(self, records: List[int]) -> int:
i, j = 0, len(records) - 1
while i <= j:
m = (i + j) // 2
if records[m] == m: i = m + 1
else: j = m - 1
return i
class Solution {
public int takeAttendance(int[] records) {
int i = 0, j = records.length - 1;
while(i <= j) {
int m = (i + j) / 2;
if(records[m] == m) i = m + 1;
else j = m - 1;
}
return i;
}
}
class Solution {
public:
int takeAttendance(vector<int>& records) {
int i = 0, j = records.size() - 1;
while(i <= j) {
int m = (i + j) / 2;
if(records[m] == m) i = m + 1;
else j = m - 1;
}
return i;
}
};
复杂度分析:
- 时间复杂度 $O(\log N)$: 二分法为对数级别复杂度。
- 空间复杂度 $O(1)$: 几个变量使用常数大小的额外空间。






