共计 5655 个字符,预计需要花费 15 分钟才能阅读完成。
第一期算法专题概述
作为第一期算法训练,其难度并不难。但是作为正常的算法学习者而言。不应该被当下所束缚,所以这里会尽可能会说到更多的方法。作为对之前算法学习的回顾。这里用到的方法会在其他的文档中做出详细的描述。
0~n-1 缺失的数字
题目描述:一个长度为 n - 1 的递增排序数组中的所有数字都是唯一的,并且每个数字都在范围 0~n- 1 之内。在范围 0~n- 1 内的 n 个数字中有且只有一个数字不在该数组中,请找出这个数字。
示例一:
输入:[0,1,3]
输出:2
示例二:
输入:[0,1,2,3,4,5,6,7,9]
输出:8
限制:
1 <= 数组长度 <= 10000
思路:
一个一个找,直到 nums[i] != i; 即可。
这里一个一个找也是有不同的方式的:
- 单纯的用 if 判别;
- 按位”异或“操作;
- 算出 nums 数组中所有数字的总和,然后一个一个的减。最后减完了,直接返回即可;
代码会在后面给出。
但是这里想说的思路是一个 二分法 的思路;
我们不想一个一个的找,而是利用有序这个天生的条件。使用二分法。
代码(代码来源于网络):
// 逐位异或操作
class Solution {
public:
int missingNumber(vector<int>& nums) {int res=nums.size();
for(int i=0;i<nums.size();i++)
{res^=nums[i];
res^=i;
}
return res;
}
};
// 用 sum 一个一个的减
class Solution {
public:
int missingNumber(vector<int>& nums) {int sum=(nums.size()*(nums.size()+1))/2;
for(int i=0;i<nums.size();i++)
{sum-=nums[i];
}
return sum;
}
};
// 二分法
int GetMissingNum(const int* num, int len)
{if(num == NULL || len <= 0)
return -1;
int left = 0; // 二分查找最左边元素下标
int right = len-1; // 二分查找最右边元素下标
while (left <= right) // 查找过程
{int mid = (left + right) >> 1; // 中间元素下标
if(num[mid] != mid) // 表示 mid 处或其左边有元素缺失
{
// 若 mid == 0 或 mid 前一个元素数字和下标相等,则 mid 即为所求
if(mid == 0 || num[mid-1] == mid - 1)
return mid;
right = mid - 1; //right 减小,往左缩小范围,继续查找
}
else // 往右半部分查找
{left = mid + 1;}
}
if (left == len) // 查找到最后一个元素,其即为所求
return len;
return -1; // 无效输入,返回 -1
}
最长连续递增序列
描述:给定一个未经排序的整数数组,找到最长且 连续 的的递增序列,并返回该序列的长度。
示例一:
输入: [1,3,5,4,7]
输出: 3
解释: 最长连续递增序列是 [1,3,5], 长度为 3。尽管 [1,3,5,7] 也是升序的子序列, 但它不是连续的,因为 5 和 7 在原数组里被 4 隔开。
示例二:
输入: [2,2,2,2,2]
输出: 1
解释: 最长连续递增序列是 [2], 长度为 1。
限制:数组长度不会超过 10000。
思路:
这里给出两个思路:
- 滑动窗口:挨个找每个元素的滑动窗口。
- 动态规划:之后专门文档讲解动态规划的种种。有关本题目的动态规划相对容易,不需要显示 dp 数组只需要给结果就行。
代码:
// 双指针
class Solution {
public: // 滑动窗口
int findLengthOfLCIS(vector<int>& nums) {if(nums.empty()) return 0;
int n=nums.size(),ans=0,ancher=0;
for(int i=0;i<n;i++){if(nums[i-1]>=nums[i]){ancher=i; // 更改一次滑动窗口}else{ans=max(res,i-ancher+1); // 更改滑动窗口时,更新 res
}
}
return ans;
}
};
// 动态规划:class Solution {
public:
int findLengthOfLCIS(vector<int>& nums) {if(nums.empty()) return 0;
int n = nums.size(), cnt = 1, ans = 1;
if(n<=1) return n;
for(int i=1;i<n;i++)
{if(nums[i]>nums[i-1])
cnt++;
else cnt = 1;
ans = max(ans, cnt);
}
return ans;
}
};
// 双指针:class Solution {
public:
int findLengthOfLCIS(vector<int>& nums) {if(nums.empty()) return 0;
int n=nums.size(),first=0,second,ans=1;
for(second=1;second<n;) // 后面的那个指针遍历到最后
{
int tem=1;
if(nums[second]>nums[second-1]) // 固定一次前指针,得到一个有序的个数
{
second++;
tem=second-first; // 固定一次前指针,得到最大的个数
ans=max(ans,tem);
}
else // 当不在大于前一个的时候,{
first=second; // 把慢指针指向当前数值
second++; // 快指针往前
ans=max(ans,tem); // 比较当前最大值和之前答案取最大
tem=1;
}
}
return ans;
}
};
合并两个有序数组
描述:
给你两个有序整数数组 nums1 和 nums2,请你将 nums2 合并到 nums1 中 , 使 nums1 成为一个有序数组。
说明:
- 初始化 nums1 和 nums2 的元素数量分别为 m 和 n。
- 你可以假设 nums1 有足够的空间(空间大小大于或等于 m + n)来保存 nums2 中的元素。
示例:
输入:
nums1 = [1,2,3,0,0,0], m = 3
nums2 = [2,5,6], n = 3
输出: [1,2,2,3,5,6]
思路:
代码:
class Solution {
public:
void merge(vector<int>& nums1, int m, vector<int>& nums2, int n) {
int i = m - 1; // 作为 nums1 数组的指针
int j = n - 1; // 作为 nums2 数组的指针
int p = m + n - 1; // 最终的 nums1 数组的位置指针
while(j >= 0 && i >= 0){if(nums2[j] > nums1[i]){nums1[p--] = nums2[j--];
}
else{nums1[p--] = nums1[i--];
}
}
while(i >= 0){nums1[p--] = nums1[i--];
}
while(j >= 0){nums1[p--] = nums2[j--];
}
}
};
公交站之间的距离
描述:
环形公交路线上有 n 个站,按次序从 0 到 n – 1 进行编号。我们已知每一对相邻公交站之间的距离,distance[i] 表示编号为 i 的车站和编号为 (i + 1) % n 的车站之间的距离。
环线上的公交车都可以按顺时针和逆时针的方向行驶。
返回乘客从出发点 start 到目的地 destination 之间的最短距离。
示例 1:
输入:distance = [1,2,3,4], start = 0, destination = 1
输出:1
解释:公交站 0 和 1 之间的距离是 1 或 9,最小值是 1。
示例 2:
输入:distance = [1,2,3,4], start = 0, destination = 2
输出:3
解释:公交站 0 和 2 之间的距离是 3 或 7,最小值是 3。
示例 3:
~~~~
输入:distance = [1,2,3,4], start = 0, destination = 3
输出:4
解释:公交站 0 和 3 之间的距离是 6 或 4,最小值是 4。
限制:
- 1 <= n <= 10^4
- distance.length == n
- 0 <= start, destination < n
- 0 <= distance[i] <= 10^4
思路:
和补码的想法类似:sum 是一定的,而言只有一个环,那么情况就是直接的距离 d 还是反绕一圈的距离 sum- d 哪一个更小。
代码:
class Solution {
public:
int distanceBetweenBusStops(vector<int>& distance, int start, int destination) {
int sum = 0;
int a = 0,i;
for(int i = 0; i < distance.size(); i++)
{sum+=distance[i];
}
for(i = start>destination?destination:start ; i < (start>destination?start:destination); i++)
{a += distance[i];
}
return a>(sum-a)?(sum-a):a;
}
};
删除数组中的重复项
描述:
给定一个排序数组,你需要在 原地 删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度。
不要使用额外的数组空间,你必须在 原地 修改输入数组 并在使用 O(1) 额外空间的条件下完成。
示例 1:
给定数组 nums = [1,1,2],
函数应该返回新的长度 2, 并且原数组 nums 的前两个元素被修改为 1, 2。
你不需要考虑数组中超出新长度后面的元素。
示例 2:
给定 nums = [0,0,1,1,1,2,2,3,3,4],
函数应该返回新的长度 5, 并且原数组 nums 的前五个元素被修改为 0, 1, 2, 3, 4。
你不需要考虑数组中超出新长度后面的元素。
说明:
为什么返回数值是整数,但输出的答案是数组呢?
请注意,输入数组是以「引用」方式传递的,这意味着在函数里修改输入数组对于调用者是可见的。
你可以想象内部操作如下:
// nums 是以“引用”方式传递的。也就是说,不对实参做任何拷贝
int len = removeDuplicates(nums);
// 在函数里修改输入数组对于调用者是可见的。
// 根据你的函数返回的长度, 它会打印出数组中该长度范围内的所有元素。
for (int i = 0; i < len; i++) {
print(nums[i]);
}
思路:
首先看说明,修改操作对数组调用者是可见的。这就意味的别做直接把元素拿出扔进 set 里面,再返回结果这种偷鸡摸狗的操作,乖乖自己写。
思路 1:
其问题的核心思想是扫描和替换。仍然可以使用双指针操作。
思路 2:
用 index 记录不重复数字的个数。
如果 nums[i] != nums[i-1]. 则出现了新的不重复值,令 nums[index] = nums[i]即可
值得注意的是:第一个数字 nums[0]肯定要记录下来。所以判断条件是 i - 1 不是 i +1. 而且 i + 1 还要作数组的判断。
思路 3:
在思路 2 里面当不相等时则判断有不重复数字。但是数字是递增的,所以只有后一个 > 前一个也是出现了新的不重复数字。其优势在于解决了思路 2 的特殊情况,即空数组情况。
代码:
// 双指针(Java 也能实现指针思路):public class Solution {public int replaceTwo(int[] nums){if(nums.length <= 2) {return nums.length;}
//1,1,1,2,2,3
int count = 1;
//count 为元素重复次数
int k = 1;
// 核心思想是替换,将元素重复次数大于 2 的部分忽略,将其他合法元素移动到前面合法区域。// 这里用 i 和 k 同时进行扫描,但是,当元素 count 不小于 2 时,(即两数相等次数达到 2,也就是说相同元素达到 3 时)
// 这时,k 会停下,直到 i 找到下一个与当前 k - 1 指向当元素不相等的那一个坐标,才继续进行替换,重复所有操作。for(int i = 1; i < nums.length; i++){
// 如果相等,则重复次数加 1
if(nums[i] == nums[i-1]) {if(count < 2) {
// 当重复达到两次时, 才进行变换操作
nums[k++] = nums[i];
count++;
}
}else {
// 如果遇到两数不想等。则情况 count 次数
count = 1;
nums[k++] = nums[i];
}
}
return k;
}
public static void main(String[] args) {Solution sl = new Solution();
int nums1[] = {0,0,1,1,1,2,3,3};
int nums2[] = {0,0,1,1,1,2,3,3};
int len1 = sl.replaceTwo(nums1);
int len2 = sl.replaceTwo(nums2);
for(int i = 0;i < len1; i++){System.out.print(nums1[i] + " ");
}
System.out.println();
for(int j = 0;j < len2; j++){System.out.print(nums2[j] + " ");
}
}
}
// 思路 2
class Solution {
public:
int removeDuplicates(vector<int>& nums) {int n = nums.size();
if (n == 0) {return 0;}
int index = 0;
// nums[0]不变,index 从 1 开始赋值。for (auto i = 1; i < n; ++ i) {if (nums[i] != nums[i - 1]) {
index ++;
nums[index] = nums[i];
}
}
return index + 1;
}
};
// 思路 3
class Solution {
public:
int removeDuplicates(vector<int>& nums) {int i = !nums.empty();
for (int n : nums)
if (n > nums[i - 1])
nums[i ++] = n;
return i;
}
};