参考引用:代码随想录
- 注:每道 LeetCode 题目都使用 ACM 代码模式,可直接在本地运行,蓝色字体为题目超链接
704. 二分查找
给定一个 n 个元素有序的(升序)整型数组 nums 和一个目标值 target ,写一个函数搜索 nums 中的 target,如果目标值存在返回下标,否则返回 -1
- 示例 1
输入: nums = [-1,0,3,5,9,12],target = 9
输出: 4
解释: 9 出现在 nums 中并且下标为 4- 示例 2
输入: nums = [-1,0,3,5,9,12],target = 2
输出: -1
解释: 2 不存在 nums 中因此返回 -1- 提示
你可以假设 nums 中的所有元素是不重复的
n 将在 [1, 10000] 之间
nums 的每个元素都将在 [-9999, 9999] 之间
使用二分法的前提条件
二分法核心思想
// 时间复杂度:O(log n)
// 空间复杂度:O(1)
#include
#include
class Solution {
public:
int search(std::vector<int> &nums, int target) {
int left = 0;
int right = nums.size() - 1; // 定义target在左闭右闭的区间里,[left, right]
while (left <= right) { // 当left==right,区间[left, right]依然有效,所以用 <=
int middle = left + ((right - left) / 2); // 防止溢出 等同于(left + right)/2
if (nums[middle] > target) {
right = middle - 1; // target 在左区间,所以[left, middle - 1]
} else if (nums[middle] < target) {
left = middle + 1; // target 在右区间,所以[middle + 1, right]
} else { // nums[middle] == target
return middle; // 数组中找到目标值,直接返回下标
}
}
// 未找到目标值
return -1;
}
};
int main(int argc, char *argv[]) {
std::vector<int> nums{ 1, 2, 3, 4, 7, 9, 10 };
int target = 2;
Solution solution;
int result = solution.search(nums, target);
std::cout << result << std::endl;
return 0;
}
// 时间复杂度:O(log n)
// 空间复杂度:O(1)
#include
#include
class Solution {
public:
int search(std::vector<int> &nums, int target) {
int left = 0;
int right = nums.size(); // 定义target在左闭右开的区间里,即:[left, right)
while (left < right) { // 因为left == right的时候,在[left, right)是无效的空间,所以使用 <
int middle = left + (right - left) / 2;
if (nums[middle] > target) {
right = middle;
} else if (nums[middle] < target) {
left = middle + 1;
} else {
return middle;
}
}
return -1;
}
};
int main(int argc, char *argv[]) {
std::vector<int> nums{ 1, 2, 3, 4, 7, 9, 10 };
int target = 2;
Solution solution;
int result = solution.search(nums, target);
std::cout << result << std::endl;
return 0;
}
27. 移除元素
给你一个数组 nums 和一个值 val,你需要原地移除所有数值等于 val 的元素,并返回移除后数组的新长度。不要使用额外的数组空间,你必须仅使用 O ( 1 ) O(1) O(1) 额外空间并原地修改输入数组
- 示例 1
给定 nums = [3,2,2,3], val = 3, 函数应该返回新的长度 2, 并且 nums 中的前两个元素均为 2- 示例 2
给定 nums = [0,1,2,2,3,0,4,2], val = 2, 函数应该返回新的长度 5, 并且 nums 中的前五个元素为 0, 1, 3, 0, 4- 提示
元素的顺序可以改变
你不需要考虑数组中超出新长度后面的元素
数组的元素在内存地址中是连续的,不能单独删除数组中的某个元素,只能覆盖
数组在内存中是连续的地址空间,不能释放单一元素,如果要释放,就是全释放(程序运行结束,回收内存栈空间)
// 时间复杂度:O(n^2)
// 空间复杂度:O(1)
#include
#include
class Solution {
public:
int removeElement(std::vector<int> &nums, int val) {
int size = nums.size();
for (int i = 0; i < size; ++i) {
if (nums[i] == val) {
// 发现需要移除的元素,就将数组集体向前移动一位
for (int j = i + 1; j < size; ++j) {
nums[j - 1] = nums[j];
}
// 因为下标 i 以后的数值都向前移动了一位,所以如果不对 i 进行自减操作
// 那么下次循环时会漏掉移动后的当前位置(即原来的下标 i+1)
// 从而导致这个位置上的元素没有被处理到
--i;
--size; // 此时数组的大小 -1
}
}
return size;
}
};
int main(int argc, char *argv[]) {
std::vector<int> nums{ 0, 1, 2, 2, 3, 0, 4, 2 };
Solution solution;
int result = solution.removeElement(nums, 2);
std::cout << result << std::endl;
return 0;
}
// 时间复杂度:O(n)
// 空间复杂度:O(1)
#include
#include
class Solution {
public:
int removeElement(std::vector<int> &nums, int val) {
int slowIndex = 0;
for (int fastIndex = 0; fastIndex < nums.size(); ++fastIndex) {
if (val != nums[fastIndex]) {
nums[slowIndex] = nums[fastIndex];
++slowIndex;
}
}
return slowIndex;
}
};
int main(int argc, char *argv[]) {
std::vector<int> nums{ 0, 1, 2, 2, 3, 0, 4, 2 };
Solution solution;
int result = solution.removeElement(nums, 2);
std::cout << result << std::endl;
return 0;
}
977. 有序数组的平方
给你一个按非递减顺序排序的整数数组 nums,返回每个数字的平方组成的新数组,要求也按非递减顺序排序
- 示例 1
输入:nums = [-4,-1,0,3,10] 输出:[0,1,9,16,100] 解释:平方后,数组变为 [16,1,0,9,100],排序后,数组变为 [0,1,9,16,100]- 示例 2
输入:nums = [-7,-3,2,3,11] 输出:[4,9,9,49,121]
// 时间复杂度:O(n + nlogn)
#include
#include
#include
class Solution {
public:
std::vector<int> sortedSquares(std::vector<int> &A) {
for (int i = 0; i < A.size(); ++i) {
A[i] *= A[i];
}
sort(A.begin(), A.end()); // 快速排序
return A;
}
};
int main(int argc, char *argv[]) {
std::vector<int> A{ -7, -3, 2, 3, 11 };
Solution solution;
std::vector<int> result = solution.sortedSquares(A); // 调用快速排序算法
// 打印排序后的数组
for (auto i = result.begin(); i < result.end(); ++i) {
std::cout << *i << ' ';
}
return 0;
}
如果 A[i] * A[i] < A[j] * A[j] 那么 result[k--] = A[j] * A[j];
如果 A[i] * A[i] >= A[j] * A[j] 那么 result[k--] = A[i] * A[i];
// 时间复杂度:O(n)
#include
#include
class Solution {
public:
std::vector<int> sortedSquares(std::vector<int> &A) {
std::vector<int> result(A.size());
// 定义左右指针,初始时分别指向数组的第一个和最后一个元素
int left = 0;
int right = A.size() - 1;
// 从右向左遍历数组
for (int i = A.size() - 1; i >= 0; --i) {
// 取左右指针指向的数的平方的较大值添加到结果中,并将左右指针向中间移动
if (A[left] * A[left] >= A[right] * A[right]) {
result[i] = A[left] * A[left];
++left;
} else {
result[i] = A[right] * A[right];
--right;
}
}
return result;
}
};
int main(int argc, char *argv[]) {
std::vector<int> A{ -7, -3, 2, 3, 11 };
Solution solution;
std::vector<int> result = solution.sortedSquares(A); // 调用快速排序算法
// 打印排序后的数组
for (auto i = result.begin(); i < result.end(); ++i) {
std::cout << *i << ' ';
}
return 0;
}
209. 长度最小的子数组
给定一个含有 n 个正整数的数组和一个正整数 s ,找出该数组中满足其和 ≥ s 的长度最小的连续子数组,并返回其长度。如果不存在符合条件的子数组,返回 0
- 示例 1
输入:s = 7, nums = [2,3,1,2,4,3] 输出:2 解释:子数组 [4,3] 是该条件下的长度最小的子数组- 提示
1 <= target <= 10^9
1 <= nums.length <= 10^5
1 <= nums[i] <= 10^5
所谓滑动窗口(也可以理解为双指针法的一种),就是不断的调节子序列的起始位置和终止位置,从而得出要想的结果。在暴力解法中,是一个 for 循环为滑动窗口的起始位置,另一个 for 循环为滑动窗口的终止位置,用两个 for 循环 完成了一个不断搜索区间的过程。那么滑动窗口如何用一个 for 循环来完成这个操作呢?
在本题中实现滑动窗口,主要确定如下三点
解题的关键在于窗口的起始位置如何移动,滑动窗口的精妙之处在于,根据当前子序列和大小的情况,不断调节子序列的起始位置,从而将 O ( n 2 ) O(n^2) O(n2) 暴力解法降为 O ( n ) O(n) O(n):每个元素在滑动窗后进来操作一次,出去操作一次,每个元素都是被操作两次,所以时间复杂度是 2n 也就是 O ( n ) O(n) O(n)
// 时间复杂度:O(n)
// 空间复杂度:O(1)
#include
#include
class Solution {
public:
int minSubArrayLen(int target, std::vector<int> &nums) {
int result = INT32_MAX; // 子数组初始值
int sum = 0; // 滑动窗口数值之和
int i = 0; // 滑动窗口起始位置
int subLength = 0; // 滑动窗口的长度
for (int j = 0; j < nums.size(); ++j) { // j 滑动窗口终止位置
sum += nums[j];
// 使用 while,每次更新 i(起始位置),并不断比较子序列是否符合条件
while (sum >= target) {
subLength = (j - i + 1); // 取子序列的长度
result = result < subLength ? result : subLength;
sum -= nums[i]; // 不断变更 i(子序列的起始位置)
++i;
}
}
// 如果 result 没有被赋值的话,就返回 0,说明没有符合条件的子序列
return result == INT32_MAX ? 0 : result;
}
};
int main(int argc, char *argv[]) {
std::vector<int> nums{ 2, 3, 1, 2, 4, 3 };
int target = 7;
Solution solution;
std::cout << solution.minSubArrayLen(target, nums) << std::endl;
return 0;
}
59. 螺旋矩阵 II
给定一个正整数 n,生成一个包含 1 到 n 2 n^2 n2 所有元素,且元素按顺时针顺序螺旋排列的正方形矩阵
- 示例
输入: 3 输出: [ [ 1, 2, 3 ], [ 8, 9, 4 ], [ 7, 6, 5 ] ]
// 时间复杂度 O(n^2): 模拟遍历二维矩阵的时间
// 空间复杂度 O(1)
#include
#include
class Solution {
public:
std::vector<std::vector<int>> generateMatrix(int n) {
std::vector<std::vector<int>> res(n, std::vector<int>(n, 0)); // 使用vector定义一个二维数组,每个位置初始化为0
int startx = 0, starty = 0; // 定义每循环一个圈的起始位置
int loop = n / 2; // 每个圈循环几次,例如n为奇数3,那么loop = 1 只是循环一圈,矩阵中间的值需要单独处理
int mid = n / 2; // 矩阵中间的位置,例如:n为3,中间的位置就是(1,1),n为5,中间位置为(2, 2)
int count = 1; // 用来给矩阵中每一个空格赋值
int offset = 1; // 需要控制每一条边遍历的长度,每次循环右边界收缩一位
int i,j;
while (loop--) {
i = startx;
j = starty;
// 下面开始的四个for就是模拟转了一圈
// 模拟填充上行从左到右(左闭右开)
for (j = starty; j < n - offset; j++) {
res[i][j] = count++;
}
// 模拟填充右列从上到下(左闭右开)
for (i = startx; i < n - offset; i++) {
res[i][j] = count++;
}
// 模拟填充下行从右到左(左闭右开)
for (; j > starty; j--) {
res[i][j] = count++;
}
// 模拟填充左列从下到上(左闭右开)
for (; i > startx; i--) {
res[i][j] = count++;
}
// 第二圈开始的时候,起始位置要各自加1, 例如:第一圈起始位置是(0, 0),第二圈起始位置是(1, 1)
startx++;
starty++;
// offset 控制每一圈里每一条边遍历的长度
offset += 1;
}
// 如果n为奇数的话,需要单独给矩阵最中间的位置赋值
if (n % 2) {
res[mid][mid] = count;
}
return res;
}
};
int main(int argc, char *argv[]) {
Solution solution;
int n = 3;
std::vector<std::vector<int>> res = solution.generateMatrix(n);
for (int i = 0; i < n; ++i) {
for (int j = 0; j < n; ++j) {
std::cout << res[i][j] << " ";
}
std::cout << std::endl;
}
return 0;
}