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

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

1. LeetCode 977.有序数组的平方

1.1 自己的思路:

  1. 将数组每个元素平方
  2. 然后冒泡排序
  3. 但怕这样效率太低过不了,但还是过了

1.2 代码

class Solution {
    public int[] sortedSquares(int[] nums) {
        //将数组元素平方
        for(int i=0;i<nums.length;i++){
            nums[i]=nums[i]*nums[i];
        }
        //冒泡排序
        for(int i=0;i<nums.length-1;i++){
            for(int j=0;j<nums.length-1-i;j++){
                if(nums[j]>nums[j+1]){
                    int temp=nums[j];
                    nums[j]=nums[j+1];
                    nums[j+1]=temp;
                }
            }
        }
        return nums;
    }
}

1.3 双指针思路

  1. 为什么想到的呢?
  2. 因为对于题目的数组中的元素平方后,一定会呈现于两边大中间小的情况,因为有负数嘛,这样可以用两个指针逐步向中间靠拢得到一个由大到小的数组
  3. 创建一个新数组result,k=nums.length-1,元素从右往左更新
  4. 定义左指针(i)右指针(j)放入循环中,只有左指针在右指针左边(i<=j)就继续循环,为什么要有=呢?假如两指针相遇时如果没有=就退出循环了,就会落下了那个元素就没有更新
  5. i什么时候++,取决于最大值是不是在i那个位置,j什么时候--取决于那个值是否要取j那个位置的元素
  6. 比较i和j两个位置的值哪个大,把大的放入result的k位置,然后k--,取了i位置的元素i就++,取了j位置的元素j就--

1.4 代码

class Solution {
    public int[] sortedSquares(int[] nums) {
        //创建一个新的数组,长度和原来一样
        int[] result=new int[nums.length];
        //定义一个指针k,从右往左给result添加元素
        int k=nums.length-1;
        //定义左指针i,右指针j,只要左指针在右指针左边包括相遇都继续执行
        for(int i=0,j=nums.length-1;i<=j;){//为什么要<=呢?因为没有等于的话相遇后就落下了相遇的元素
            if(nums[i]*nums[i]>nums[j]*nums[j]){//左指针的元素平方后更大
                result[k--]=nums[i]*nums[i++];//将平方后元素放入,然后k--,i++
            }else{//右指针的元素平方后更大
                result[k--]=nums[j]*nums[j--];//将平方后元素放入,然后k--,j--
            }
        }
        return result;
    }
}

2. LeetCode 209.长度最小的子数组

2.1 自己的思路

       说实话,没思路,我开始以为会是贪心,但贪心我也不会啊

2.2 滑动窗口(双指针)思路

  1. j表示滑动窗口的起始还是终止位置呢?可以假设是起始,但如果是这样的话那么求和时终止位置就要把j后面所有的元素都遍历一遍才能又回到j这个位置,这样跟暴力求解思路是一样的,所以j应该表示终止位置
  2. 关键在如何移动起始位置,j随着for循环后移,当sum大于等于target时起始位置就可以移动了,当sum大于等于target时滑动窗口就形成了
  3. 形成滑动窗口后就求窗口的长度,subL=j-i+1,result应该在result和subL中取个最小值作为长度最小的子数组,然后sum总和要减去nums[i]这个起始位置的值
  4. 中间条件(sum>=target)前是写if还是while呢?举个例子:假如数组[1,1,1,100,...],target=100,只有[1,1,1,100]这部分符合大于等于target的,如果写if,那么只去掉了一个1,还没有达到最短,只有写while才能求得最后的子数组长度为1,也就是[100]这个数组

2.3 代码

class Solution {
    public int minSubArrayLen(int target, int[] nums) {
        //i为起始位置,左指针
        int i=0;
        //将长度设置为数组长度+1
        int result=nums.length+1;
        int sum=0;
        //j为终止位置,右指针
        for(int j=0;j<nums.length;j++){
            //先求sum,从起始到终止的和
            sum+=nums[j];
            while(sum>=target){
                //在result中和j-1+1取个最小值
                result=Math.min(result,j-i+1);
                //sum减去起始位置的值,然后起始位置后移
                sum-=nums[i++];
            }
        }
        //如果长度仍为数组长度+1,则说明没有符合target条件,则返回0
        return result==(nums.length+1)?0:result;
    }
}

3. LeetCode 59. 螺旋矩阵 II

3.1 自己的思路

       尝试过把图画出后找找看能不能发现数学上的规律,结果不能

3.2 二分法思路

  1. 遍历每条边的规则是左闭右开还是左闭右闭呢?按道理都可以
  2. 如果是按照左闭右开的规则呢?处理最上面的边的时候从左至右除了最后一个节点就都处理了,就是处理每一条边的时候都只是处理第一个节点到最后倒数第二个节点也就是最后一个节点不处理,留给下一条边
  3. 我们定义loop为要转的圈数,loop=n/2的,可以画个正方形的图,然后按照题意模拟转圈就可以知道要转多少圈了
  4. 我们定义startX、startY为0作为每一圈的起始位置,到下一圈了就++
  5. 我们定义offset=1来控制每条边遍历的长度,每转了一圈就++,含义就是从左至右以及从上至下遍历的时候距离边界的距离为1
  6. 当n为奇数时按照我们上面的定义是少转了一圈,差了中间位置的值,因此定义mid=n/2,表示中间位置,循环结束后直接赋值

3.3 代码

class Solution {
    public int[][] generateMatrix(int n) {
        int[][] res=new int[n][n];
        int startx = 0, starty = 0; // 作为每一圈的起始位置,到下一圈了就++
        int loop = n / 2; // 表示要转的圈数
        int mid = n / 2; // 矩阵中间的位置
        int count = 1; 
        int offset = 1; // 控制每条边遍历的长度,每转了一圈就++
        int i,j;
        while (loop!=0) {
            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
            startx++;
            starty++;
            // offset 控制每一圈里每一条边遍历的长度,每转一圈就++,距离边界的距离+1
            offset++;
            //转的圈数-1
            loop--;
        }
        // 如果n为奇数的话,需要单独给矩阵最中间的位置赋值
        if (n % 2==1) {
            res[mid][mid] = count;
        }
        return res;
    }
}

4. 数组总结

相关文章
|
1月前
leetCode(删除有序数组中的重复项)
如何在不使用额外空间的情况下,通过双指针法原地删除有序数组中的重复项。
33 2
|
1月前
【LeetCode 48】108.将有序数组转换为二叉搜索树
【LeetCode 48】108.将有序数组转换为二叉搜索树
38 0
|
3月前
|
算法
LeetCode第80题删除有序数组中的重复项 II
文章介绍了LeetCode第80题"删除有序数组中的重复项 II"的解法,利用双指针技术在O(1)空间复杂度内原地删除重复元素,并总结了双指针技术在处理有序数组问题中的应用。
LeetCode第80题删除有序数组中的重复项 II
|
3月前
|
算法
LeetCode第88题合并两个有序数组
文章分享了LeetCode第88题"合并两个有序数组"的解法,通过从后向前的合并策略避免了数组元素的前移,使用三个指针高效地完成了合并过程。
|
2月前
|
Unix Shell Linux
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
本文提供了几个Linux shell脚本编程问题的解决方案,包括转置文件内容、统计词频、验证有效电话号码和提取文件的第十行,每个问题都给出了至少一种实现方法。
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
|
3月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
56 6
|
3月前
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
113 2
|
17天前
|
机器学习/深度学习 人工智能 自然语言处理
280页PDF,全方位评估OpenAI o1,Leetcode刷题准确率竟这么高
【10月更文挑战第24天】近年来,OpenAI的o1模型在大型语言模型(LLMs)中脱颖而出,展现出卓越的推理能力和知识整合能力。基于Transformer架构,o1模型采用了链式思维和强化学习等先进技术,显著提升了其在编程竞赛、医学影像报告生成、数学问题解决、自然语言推理和芯片设计等领域的表现。本文将全面评估o1模型的性能及其对AI研究和应用的潜在影响。
16 1
|
2月前
|
数据采集 负载均衡 安全
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
本文提供了多个多线程编程问题的解决方案,包括设计有限阻塞队列、多线程网页爬虫、红绿灯路口等,每个问题都给出了至少一种实现方法,涵盖了互斥锁、条件变量、信号量等线程同步机制的使用。
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
|
3月前
|
索引 Python
【Leetcode刷题Python】从列表list中创建一颗二叉树
本文介绍了如何使用Python递归函数从列表中创建二叉树,其中每个节点的左右子节点索引分别是当前节点索引的2倍加1和2倍加2。
56 7