当前位置: 首页 > article >正文

day002-数组-有序数组的平方、长度最小的子数组、螺旋矩阵II

977.有序数组的平方

题目建议: 本题关键在于理解双指针思想

题目链接:https://leetcode.cn/problems/squares-of-a-sorted-array/
文章讲解:https://programmercarl.com/0977.%E6%9C%89%E5%BA%8F%E6%95%B0%E7%BB%84%E7%9A%84%E5%B9%B3%E6%96%B9.html
视频讲解: https://www.bilibili.com/video/BV1QB4y1D7ep

class Solution {
public:
    vector<int> sortedSquares(vector<int>& nums) {
        vector<int> result(nums.size(), 0);
        int low = 0, high = nums.size() - 1;
        int idx = high;
        while (low <= high) {
            if (nums[low] * nums[low] >= nums[high] * nums[high]) {
                result[idx--] = nums[low] * nums[low];
                low++;
            }
            else {
                result[idx--] = nums[high] * nums[high];
                high--;
            }
        }
        return result;
    }
};

思路

看到题目首先想到的是暴力遍历,但是效率太低,是O(n + log n), 第一个n是算平方,第二个logn是排序,是不可接受的,不满足题目要求。所以需要用双指针法,因为原数组是升序的,只有头尾可能有最大值,所以需要头尾双指针向内收缩,比较后插值,
时间复杂度为O(n),
空间复杂度为O(n)

注意

其中有几个遗忘的点在于vector的初始化可以通过初始函数实现,nums(size,0),其中size是元素数量,0为设定的初始值;以及c++中vector的末尾元素添加用的是push_back(), 不是python的append函数

209.长度最小的子数组

题目建议: 本题关键在于理解滑动窗口,这个滑动窗口看文字讲解 还挺难理解的,建议大家先看视频讲解。 拓展题目可以先不做。

题目链接:
文章讲解:
视频讲解:

class Solution {
public:
// 这个双指针滑动窗口的意义是在每一个右边界时找到最大左边界,即子数组结尾元素固定时,找最短的头元素
    int minSubArrayLen(int target, vector<int>& nums) {
        // int left, right, sum = 0, result = INT_MAX;
        // for (left = 0, right = 0; right < nums.size();right++) {
        //     sum += nums[right];
        //     while (sum >= target) {
        //         result = result < (right - left + 1) ? result : (right - left + 1);
        //         sum -= nums[left];
        //         left++;
        //     }
            
        // }
        // return result == INT_MAX ? 0 : result;

        int left = 0, right = 1, sum = nums[0], result = INT_MAX;
        for (right = 1; right <= nums.size();right++) {
            while (sum >= target) {
                result = result < (right - left) ? result : (right - left);
                sum -= nums[left];
                left++;
            }
            if (right < nums.size()) {
                sum += nums[right];
            }
            
        }
        return result == INT_MAX ? 0 : result;
    }
};

思路

第一个想到的是双侧循环,暴力求解,但时间复杂度是O(n^2),题目要求O(n),所以不可行,所以用双指针滑动窗口,而由于该题中的原数组是无序的,所以在每次改变窗口大小时,需要右一个固定边界,以固定边界作为遍历的标志。在这里需要求最小子数组长度,所以以右侧边界为遍历标志。
实质上本题的双指针窗口的意义是在每一个右边界时找到最大左边界,即子数组结尾元素固定时,找最短的头元素
时间复杂度O(n)
空间复杂度O(1)

注意

尾减头长度计算需要记得+1,整数最大值是INT_MAX,三目条件运算符格式例如取最大:x= x>y?x:y
如果头尾两个while循环,是需要数组事先排序,如升序排序下,和大了就缩小左边界,和小了就放大右边界。

59.螺旋矩阵II

题目建议:本题关键还是在转圈的逻辑,在二分搜索中提到的区间定义,在这里又用上了。

题目链接:
文章讲解:
视频讲解:

自己写的

class Solution {
public:
    vector<vector<int>> generateMatrix(int n) {
        vector<vector<int>> res(n, vector<int>(n, 0));
        int count = 1, i, j, loop;
        for (i = 0, j = 0, loop = 0; loop < n / 2; i++, j++, loop++) {
            while (j < n - 1 - loop) res[i][j++] = count++;
            while (i < n - 1 - loop) res[i++][j] = count++;
            while (j > loop) res[i][j--] = count++;
            while (i > loop) res[i--][j] = count++;
        }
        if (n % 2) res[loop][loop] = count;
        return res;
    }
};

代码随想录给的

class Solution {
public:
    vector<vector<int>> generateMatrix(int n) {
        vector<vector<int>> res(n, vector<int>(n, 0));
        int startx = 0, starty = 0;
        int loop = n / 2;
        int mid = n / 2;
        int count = 1;
        int offset = 1;
        int i, j;
        while (loop --) {
            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++;
            }
            startx++;
            starty++;
            offset++;
        }
        if (n % 2) {
            res[mid][mid] = count;
        }
        return res;
    }
};

评论老哥给的

class Solution {
public:
    vector<vector<int>> generateMatrix(int n) {
        int t = 0;      // top
        int b = n-1;    // bottom
        int l = 0;      // left
        int r = n-1;    // right
        vector<vector<int>> ans(n,vector<int>(n));
        int k=1;
        while(k<=n*n){
            for(int i=l;i<=r;++i,++k) ans[t][i] = k;
            ++t;
            for(int i=t;i<=b;++i,++k) ans[i][r] = k;
            --r;
            for(int i=r;i>=l;--i,++k) ans[b][i] = k;
            --b;
            for(int i=b;i>=t;--i,++k) ans[i][l] = k;
            ++l;
        }
        return ans;
    }
};

思路

最简单的一道题竟然搞了非常久的时间,主要的原因在于除以2的时候,应当是 / 2,顺手用了python的 // 2,想整除,结果忘了在C++里面是注释,而颜色又不显眼,导致始终没发觉错误,耽搁了非常长的时间,以为是夹杂了哪个中文全角符号,排查了很久。
代码随想录的代码,和评论老哥的代码可以看做是两个习惯了,第三种代码的习惯更贴近我的思想,只是我想尽可能节省自定义符号,所以用了loop来写,需要稍微绕一点点,也无法避免最后奇数情况下对中央的补充,所以还是第三种好一些。
当然第二种代码思路非常清晰,每次循环的起点,变量与不变量,过程,循环次数很分明,可以作为思想模板。
时间复杂度O(n^2)
空间复杂度O(1)

注意

vector建立双层数组:
vector<vector> result(n, vector(n, 0));
c++注释是 // 不要顺手用了
对于边界,还是建立专门的变量保存比较方便控制。


http://www.kler.cn/a/7739.html

相关文章:

  • Windows 11 上配置VSCode 使用 Git 和 SSH 完整步骤
  • Ubuntu 20.04安装gcc
  • 详细全面讲解C++中重载、隐藏、覆盖的区别
  • 什么是cline?
  • 使用JMeter玩转tidb压测
  • nginx-灰度发布策略(基于cookie)
  • (数字图像处理MATLAB+Python)第四章图像正交变换-第二节:离散余弦变换和K-L变换
  • CTFHub | 双写后缀
  • 使用Python、Contours绘制等高线
  • 软件安全测试有哪些测试手段?软件测试报告收费贵吗?
  • 增程汽车大厂上纯电,理想能行吗?
  • 实时聊天如何改变您的在线商店
  • 【文心一言】内测版 沉浸式深度体验——不间断 提问问题!它的表现如何?
  • SOLIDWORKS三维建模的十大应用技巧
  • 【Axure高保真原型】画图画板
  • 数组按照某个key分组
  • SpringCloud-高级篇(一)
  • 2021蓝桥杯真题小平方 C语言/C++
  • 【Java版oj】day28反转部分单向链表、猴子分桃
  • nginx 逻辑判断if语句使用
  • 【二叉树OJ题(二)】前序遍历中序遍历后序遍历另一颗树的子树二叉树遍历平衡二叉树
  • 精彩回顾 | 平行云亮相LiveVideoStack2022北京站
  • 2023年一个完整的B2B订货网站源码
  • NC65 部门预算DAO类
  • ‘protoc-gen-js‘ 不是内部或外部命令,也不是可运行的程序
  • 在DongshanPI-D1开箱使用分享与折腾记录实现MPU6050数据读取