代码随想录Day2

1,047 阅读5分钟

977.有序数组的平方

题目链接:977.有序数组的平方

文章讲解:977.有序数组的平方

视频讲解:双指针法经典题目 | LeetCode:977.有序数组的平方_哔哩哔哩_bilibili

思路:有序数组中以0为界,距离越远的元素平方值越大,也就是绝对值越大的元素平方值越大。以此思路入手,应该以左右指针分别指向左右边界通过比较来获取平方值最大的元素。设想的解法也是昨天学习的双指针法。

暴力解法

简单粗暴的先求平方值再排序。

// 暴力解法,直接求平方值再排序
    // 时间复杂度O(log n + n),空间复杂度O(1)
    vector<int> sortedSquares1(vector<int>& nums) {
        for (int i = 0; i < nums.size(); ++i) {
            // 求平方
            nums[i] *= nums[i];
        }
        // 调用标准库的sort函数,快速排序,O(log n)
        sort(nums.begin(), nums.end());
        return nums;
    }

双指针法

// 双指针法,比如-3,-2,0,5,6,平方后的最大数一定在举例0最远的左边或右边,那么左右指针分别从最远处往0触发,互相比较,即可求平方同时从大到小排序。
    // 时间复杂度O(n),空间复杂度O(1)
    vector<int> sortedSquares2(vector<int>& nums) {
        // 定义左右指针
        int left = 0;
        int right = nums.size() - 1;
        int n = right;
        // 定义一个新数组 题目要求返回一个新数组
        vector<int> res(nums.size(), 0);
        // 定义最大数插入位置下标,从后到前、从大到小插入
        // 循环条件左小于等于右,注意要有等号,不然会漏掉元素
        while (left <= right) {
            int leftSq = nums[left] * nums[left];
            int rightSq = nums[right] * nums[right];
            if (leftSq >= rightSq) {
                res[n--] = leftSq;
                ++left;
            }
            else {
                res[n--] = rightSq;
                --right;
            }
        }
        return res;
    }

209.长度最小的子数组

题目链接:209.长度最小的子数组

文章讲解:209.长度最小的子数组

视频讲解:拿下滑动窗口! | LeetCode 209 长度最小的子数组_哔哩哔哩_bilibili

思路:暴力解法当然是两层for循环,第一层的i遍历子序列的起点,第二层的j遍历子序列的终点,寻找符合要求的最短子序列。由于暴力解法的时间复杂度为O(n^2),目前在力扣已经超时无法通过,所以需要思考改进方法。

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

思考:如何使用一个for循环来解决检索问题,这个for循环应该用来表示窗口的起点还是终点? 如果用来表示起点,那么就和暴力解法思路相同了,所以应该是用来表示终点,同时思考起点如何变化。

原文中给出的动图基本上能完美诠释滑动窗口的思想:

209.长度最小的子数组.gif

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

暴力解法

// 暴力解法 O(n^2) LeetCode现在报超时了
    int minSubArrayLen1(int target, vector<int>& nums) {
        // 待返回的结果:最短子序列长度
        int res = INT32_MAX;
        // 子序列的数值之和
        int sum = 0;
        // 子序列的长度
        int length = 0;
        for (int i=0; i < nums.size(); ++i) {
            // 将sum归零
            sum = 0;
            for (int j=i; j < nums.size(); ++j) {
                sum += nums[j];
                if (sum >= target) {
                    length = j - i + 1;
                    res = res < length ? res : length;
                    //发现符合条件的子序列就跳出循环->找最短的
                    break;
                }
            }
        }
        // res依旧为INT32_MAX表示没有符合条件的子序列
        return res == INT32_MAX ? 0 : res;
    }

滑动窗口法

// 滑动窗口法
    // 时间复杂度O(n)
    int minSubArrayLen2(int target, vector<int>& nums) {
        int res = INT32_MAX;
        int length = 0;
        int sum = 0;
        // i指针指向子序列起始位置
        int i = 0;
        // j指针指向子序列末尾位置
        for (int j=0; j < nums.size(); ++j) {
            sum += nums[j];
            while (sum >= target) {
                length = j - i + 1;
                res = res < length ? res : length;
                // **重要** 起始位置移动 继续判断 使得双for循环变单for循环
                sum -= nums[i++];
            }
        }
        return res == INT32_MAX ? 0 : res;
    }

59.螺旋矩阵II

题目链接:59.螺旋矩阵II

文章讲解:59.螺旋矩阵II

视频讲解:一入循环深似海 | LeetCode:59.螺旋矩阵II_哔哩哔哩_bilibili

思路:主要就是体现一个循环不变量的理解和应用,本题使用左闭右开区间,值得一提的是,n为奇数时,中间值middle需要单独赋予,位置坐标为res[n >> 1][n >> 1],矩阵绘制需要转的圈数:loop=n >> 1,并且记录开始位置坐标的x、y值,和每次转圈的偏移量offset以便后续计算使用。

tipsc++定义二维数组:

vector<vector<int>> res(n, vector<int>(n, 0));

代码如下

// 坚持循环不变量原则,选择左闭右开
// 时间复杂度O(n^2)——>用于模拟二维矩阵
// 空间复杂度O(1)——>辅助计算所使用的内存空间才算入
    vector<vector<int>> generateMatrix(int n) {
        // 用vector定义二维数组
        vector<vector<int>> res(n, vector<int>(n, 0));
        // 定义每循环一个圈的起始位置
        int startx = 0, starty = 0;
        // 要转几个圈
        int loop = n >> 1;
        // 矩阵中间的位置
        int mid = n >> 1;
        // cout << "mid:" << mid << endl;
        // 用来给矩阵中每一个空格赋值
        int count = 1;
        // 需要控制每一条边遍历的长度,每次循环有边界收缩一位
        int offset = 1;
        int i,j;
        while (loop--) {
            i = startx;
            j = starty;

            //以下四个for模拟转一圈
            // 模拟填充上行从左到右(左闭右开)
            for (j = startx; 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;
    }

测试结果如图 image.png

数组部分总结

开局一张图系列,感觉莫法比这个图总结的更好了~ ——来自代码随想录知识星球 (opens new window)成员:海螺人

数组总结.png