首页 > 编程语言 >代码随想录算法训练营第二天|977.有序数组的平方 ,209.长度最小的子数组 ,59.螺旋矩阵II

代码随想录算法训练营第二天|977.有序数组的平方 ,209.长度最小的子数组 ,59.螺旋矩阵II

时间:2023-01-12 23:01:15浏览次数:74  
标签:977 nums int res 随想录 ++ vector 数组

一、参考资料
有序数组的平方
题目链接:https://leetcode.cn/problems/squares-of-a-sorted-array/

文章讲解:https://programmercarl.com/0977.有序数组的平方.html

视频讲解: https://www.bilibili.com/video/BV1QB4y1D7ep

长度最小的子数组
题目链接:https://leetcode.cn/problems/minimum-size-subarray-sum/

文章讲解:https://programmercarl.com/0209.长度最小的子数组.html

视频讲解:https://www.bilibili.com/video/BV1tZ4y1q7XE

螺旋矩阵II
题目链接:https://leetcode.cn/problems/spiral-matrix-ii/

文章讲解:https://programmercarl.com/0059.螺旋矩阵II.html

视频讲解:https://www.bilibili.com/video/BV1SL4y1N7mV/

数组专题总结
文章链接:https://programmercarl.com/数组总结篇.html

二、LeetCode977-有序数组的平方
我的代码(过于暴力)
class Solution {
public:
vector sortedSquares(vector& nums) {
vector res;

    for (int i = 0; i < nums.size(); i++){
        nums[i] = nums[i] * nums[i];
    }
    sort(nums.begin(), nums.end());
    return nums;
}

};
接下来学习一下双指针的代码实现:

双指针实现
class Solution {
public:
vector sortedSquares(vector& nums) {
// 这里定义一个结果集,长度为原数组的大小
vector res(nums.size(), 0);
// 定义首尾指针
int firstp = 0;
int lastp = nums.size() - 1;
int k = lastp; // k表示结果集的指针,从后向前写入vector

    while (firstp <= lastp){
        if (nums[firstp] * nums[firstp] < nums[lastp] * nums[lastp]){
            res[k--] = nums[lastp] * nums[lastp];
            lastp--;
        }
        else{
            res[k--] = nums[firstp] * nums[firstp];
            firstp++;
        }
    }
    
    return res;
}

};

三、LeetCode209-长度最小的子数组
最大的收获是学习了滑动窗口的写法!视频讲解很详细

滑动窗口-代码实现
class Solution {
public:
int minSubArrayLen(int target, vector& nums) {
int result = INT32_MAX;
int sum = 0;
// 滑动窗口的起始位置
int begin = 0;
// 滑动窗口的长度
int subLen = 0;

    for (int i = 0; i < nums.size(); i++){
        sum += nums[i];
        while (sum >= target){
            subLen = (i - begin + 1);
            result = result < subLen ? result : subLen;
            sum -= nums[begin];
            begin++;
        }
    }
    if (result == INT32_MAX){
        return 0;
    }
    
    return result;
}

};

问题:

INT32_MAX是什么?它表示数组中一个很大的数,进一步,它是在limits.h下面的一个宏。

https://blog.csdn.net/wangshuqian1314/article/details/122657716

卡哥讲解!
滑动窗口,就是不断的调节子序列的起始位置和终止位置,从而得出我们要想的结果。

滑动窗口的精妙之处在于根据当前子序列和大小的情况,不断调节子序列的起始位置。从而将O(n^2)暴力解法降为O(n)。

时间复杂度O(n),空间复杂度O(1)

class Solution {
public:
int minSubArrayLen(int s, vector& nums) {
int result = INT32_MAX;
int sum = 0; // 滑动窗口数值之和
int i = 0; // 滑动窗口起始位置
int subLength = 0; // 滑动窗口的长度
for (int j = 0; j < nums.size(); j++) {
sum += nums[j];
// 注意这里使用while,每次更新 i(起始位置),并不断比较子序列是否符合条件
while (sum >= s) {
subLength = (j - i + 1); // 取子序列的长度
result = result < subLength ? result : subLength;
sum -= nums[i++]; // 这里体现出滑动窗口的精髓之处,不断变更i(子序列的起始位置)
}
}
// 如果result没有被赋值的话,就返回0,说明没有符合条件的子序列
return result == INT32_MAX ? 0 : result;
}
};

四、LeetCode59-螺旋矩阵II
这个题以前是做过的,先用学会的模拟实现一下

我的代码(数组模拟)
class Solution {
public:
// 这题之前也写过,但还是不太熟练呐
vector<vector> generateMatrix(int n) {
vector<vector> res(n, vector(n)); // 初始化二维数组
// 定义上下左右四个方向的索引
int t = 0; // top
int b = n - 1; // bottom
int l = 0; // left
int r = n - 1; // right
// 螺旋矩阵的计数(即填入矩阵的数值)
int k = 1;
while (k <= n * n){
for(int i = l; i <= r; i++, k++){
res[t][i] = k;
}
t++;
for(int i = t; i <= b; i++, k++){
res[i][r] = k;
}
r--;
for(int i = r; i >= l; i--, k++){
res[b][i] = k;
}
b--;
for(int i = b; i >= t; i--, k++){
res[i][l] = k;
}
l++;
}
return res;
}
};

卡哥讲解!
求解本题依然是要坚持循环不变量原则【处理的原则:统一的左闭右开】

class Solution {
public:
vector<vector> generateMatrix(int n) {
vector<vector> res(n, vector(n, 0)); // 使用vector定义一个二维数组
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[startx][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;
}

};

Day02总结:

最大的收获是学习了滑动窗口,写代码的熟练度还需要加强

数组模拟注意边界问题,主要是想明白怎么轮转的,将清楚的逻辑转化为代码实现

双指针还需要多多练习,多看题目和多思考~

五、数组专题总结

标签:977,nums,int,res,随想录,++,vector,数组
From: https://www.cnblogs.com/ucaszym/p/17048199.html

相关文章

  • 代码随想录算法训练营第15天
    今日刷题3道:层序遍历, 226.翻转二叉树,101.对称二叉树2● 层序遍历10题目链接/文章讲解/视频讲解:https://programmercarl.com/0102.%E4%BA%8C%E5%8F%89%E6%A......
  • 树状数组
    树状数组树状数组是一种简洁高效的数据结构。它可以优化前缀和和差分操作。树状数组和线段树具有相似的功能,但他俩毕竟还有一些区别:树状数组能有的操作,线段树一定有;线段......
  • 工作中需知道的数组方法总结
    数组遍历操作forEach该方法等同于for循环,其没有返回值结构:arr.forEach(回调函数,回调函数this的值)第二个参数当回调函数是箭头函数时无效用法:arr.forEach(function(item,......
  • Java数组
    数组数组概述相同数据类型的数据有序集合描述相同类型的若干个数据,按照一定的先后次序排列组合一起数组里的每一个数据称作一个数组元素,每个数组元素可以通......
  • JS_3_数组
    JavaScript中的数组是自动扩容的。灵活地!  声明:1、vararr=newArray();2、vararr=newArray(length);3、vararr=[];赋值和取值:通过......
  • 1.将树形结构数据打平成一维数组
    1.函数constflatTree=(treeData)=>{letresult=[];treeData.forEach((item)=>{//先克隆一份数据作为第一层级的填充letres=JSON.parse(JSON......
  • LeetCode刷题(11)~ 两个数组的交集 II
    题目描述给定两个数组,编写一个函数来计算它们的交集。示例1:输入:nums1=[1,2,2,1],nums2=[2,2]输出:[2,2]示例2:输入:nums1=[4,9,5],nums2=[9,4,9,8,4]输出:[4,9]......
  • LeetCode刷题(8)~旋转数组
    题目描述给定一个数组,将数组中的元素向右移动k个位置,其中k是非负数。示例1:输入:[1,2,3,4,5,6,7]和k=3输出:[5,6,7,1,2,3,4]解释:向右旋转1步:[7,1,2,3,4,5,6......
  • LeetCode刷题(7)~删除排序数组中的重复项
    题目描述给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度。不要使用额外的数组空间,你必须在原地修改输入数组并在使用......
  • LeetCode刷题(59)~使数组中所有元素相等的最小操作数【第202场周赛:题目二】
    题目描述存在一个长度为n的数组arr,其中arr[i]=(2*i)+1(0<=i<n)。一次操作中,你可以选出两个下标,记作x和y(0<=x,y<n)并使arr[x]减去1、arr[y]......