59. 螺旋矩阵 II

news/2024/11/26 6:41:23/

题目链接:力扣

解题思路:类似于54. 螺旋矩阵 

找规律,模拟矩阵的访问,每访问一个位置,就对该位置进行赋值,

观察螺旋顺序:行(向右)--> 列(向下)-->行(向左)-->列(向上)-->行(向右)-->...

可以发现行和列是交替的,并且两次遍历行和列时的方向和上次访行和列问的方向正好相反

比如上次访问行是向右的,则下次再次访问行时则是向左的,下次再次访问时则是向右的,列的规律也相同

并且每次访问行和列时需要遍历的元素也是有规律的,对于m行n列的矩阵来说,假设rowNum表示已经访问过了几行,colNum表示已经访问过了几列,则对于当前正在访问的行或者列来说,需要访问的元素个数满足下列规律

  1. 对于行,当前需要访问的元素个数:n-colNum,矩阵的列数减去已经访问的列数就是当前行需要访问的元素个数,访问列的时候,一整列的元素都已经访问过了,这次访问行时访问过的列中的元素不需要再次访问
  2. 对于列,当前需要访问的元素个数:m-rowNum,矩阵的行数减去已经访问的行数就是当前列需要访问的元素个数,访问行的时候,一整行的元素已经访问过了,这次访问列时访问过的行中的元素不需要再次访问

可以使用两个变量currentRow和currentCol,表示当前正在访问的行和列,rowAdd=1和colAdd=1表示行和列的方向:

  1. 访问行时:每次令currentCol=currentCol+colAdd,直到访问n-colNum个元素为止,然后令colAdd=-colAdd(下次访问行时按照相反的方向访问),rowNum++(已经访问过的行的数量加1)
  2. 访问列时:每次令currentRow=currentRow+rowAdd,直到访问m-rowNum个元素为止,然后令rowAdd=-rowAdd(下次列时按照相反的方向访问),colNum++
  3. 无论访问行还是列,都将当前正在访问的元素matrix[currentRow][currentCol]加入到结果result中

可以使用一个标记rowFlag交替访问行和列,当result元素中的个数等于m*n时,结束

使用一个全局变量currentNum=1依次给每个访问的元素赋值,每次赋值完,自增1

AC代码:

class Solution {public static int[][] generateMatrix(int n) {int[][] result = new int[n][n];int currentNum = 1;int rowNum = 0;int colNum = 0;boolean rowFlag = true;int rowAdd = 1;int colAdd = 1;int currentRow = 0;int currentCol = -1;while (currentNum != n * n + 1) {if (rowFlag) {//遍历当前行int num = n - colNum;while (num > 0) {currentCol += colAdd;result[currentRow][currentCol] = currentNum++;num--;}rowFlag = false;colAdd = -colAdd;rowNum++;} else {//遍历当前列int num = n - rowNum;while (num > 0) {currentRow += rowAdd;result[currentRow][currentCol] = currentNum++;num--;}rowFlag = true;rowAdd = -rowAdd;colNum++;}}return result;}
}

时间复杂度O(n),空间复杂度O(n^2)

更简洁的做法:时间和空间复杂度和上面代码一样,不过代码更加简短

从外层向内层填数字,每次填一圈,填完最外层一圈,然后在填内层的一圈,每一圈填数字的顺序按照:上边界的行(从左到右)-->右边界的列(从上到下) --> 下边界的行(从右到左) -->左边界的列(从下到上)

这时需要定义四个边界,left,right,top,bottom,分别表示当前正在填写的外圈的四个边界,具体算法如下:

  1. 初始值:left=0,right = n-1,top=0,bottom=n-1,currentNum=1
  2. 当currentNum小于n*n-1时,循环对外层边界填数字:
    1. 对上边界的行填数字:
      1. result[top][i]=currentNum++;
    2. 上边界已经填写完,修改上边界:top++
    3. 对右边界的列填数字:
      1. result[i][right]=currentNum++;
    4. 右边界已经填写完,修改右边界:right--
    5. 对下边界的行填数字
      1. result[bottom][i]=currentNum++;
    6. 下边界已经填写完,修改下边界:bottom--
    7. 对左边界的列填数字
      1. result[i][left]=currentNum++
    8. 左边界已经填写完,修改左边界:left++

AC代码

class Solution {public static int[][] generateMatrix(int n) {int left = 0;int right = n - 1;int top = 0;int bottom = n - 1;int currentNum = 1;int[][] result = new int[n][n];while (currentNum < n * n+1) {//上边界,从左到右for (int i = left; i <= right; i++)result[top][i] = currentNum++;top++;//右边界,从上到下for (int i = top; i <= bottom; i++)result[i][right] = currentNum++;right--;//下边界,从右到左for (int i = right; i >= left; i--)result[bottom][i] = currentNum++;bottom--;//左边界,从下到上for (int i = bottom; i >= top; i--)result[i][left] = currentNum++;left++;}return result;}
}

 


http://www.ppmy.cn/news/974781.html

相关文章

12.面板问题

面板问题 html部分 <h1>Lorem ipsum dolor sit, amet consectetur adipisicing.</h1><div class"container"><div class"faq"><div class"title-box"><h3 class"title">Lorem, ipsum dolor.<…

linux:secureCRT通过pem证书远程访问服务器

参考&#xff1a; secureCRT通过pem证书远程访问服务器_Fengshana的博客-CSDN博客 总结&#xff1a; 配置公钥即可

“深入解析Spring Boot:从入门到精通“

标题&#xff1a;深入解析Spring Boot&#xff1a;从入门到精通 摘要&#xff1a;本文将全面介绍Spring Boot框架的基本概念、核心特性、使用方法以及提供示例代码&#xff0c;帮助读者从入门到精通Spring Boot开发。 引言 Spring Boot是一个用于快速构建基于Spring框架的应用…

Android:Service服务简单应用

public class MyService extends Service { public MyService() { System.out.println("MyServiceMyService()"); } Override public void onCreate() { super.onCreate(); System.out.println("MyService创建onCreate()&quo…

2023/7/23周报

目录 摘要 论文阅读 1、题目和现存问题 2、问题阐述及相关定义 3、LGDL模型框架 4、实验准备 5、实验过程 深度学习 1、GCN简单分类任务 2、文献引用数据分类案例 3、将时序型数据构建为图数据格式 总结 摘要 本周在论文阅读上&#xff0c;对基于图神经网络与深度…

C语言编程---案例练习

文章目录 格式化输出 格式化输出 %d&#xff0c;输出整数&#xff1b; %f&#xff0c;输出浮点数&#xff1b;%.3f 保留三位小数&#xff1b; %e&#xff0c;输出双精度浮点数&#xff1b; %c&#xff0c;输出单个字符&#xff1b;将字符格式化%d&#xff0c;即转ASCII码&…

83、讲下Zookeeper watch机制

讲下Zookeeper watch机制 客户端&#xff0c;可以通过在znode上设置watch&#xff0c;实现实时监听znode的变化。 Watch事件是一个一次性的触发器&#xff0c;当被设置了Watch的数据发生了改变的时候&#xff0c;则服务器将这个改变发送给设置了Watch的客户端 父节点的创建&…

LeetCode[1508]子数组和排序后的区间和

难度&#xff1a;Medium 题目&#xff1a; 给你一个数组 nums &#xff0c;它包含 n 个正整数。你需要计算所有非空连续子数组的和&#xff0c;并将它们按升序排序&#xff0c;得到一个新的包含 n * (n 1) / 2 个数字的数组。 请你返回在新数组中下标为 left 到 right &#…