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

leetcode--螺旋矩阵

LCR 146.螺旋遍历二维数组

给定一个二维数组 array,请返回「螺旋遍历」该数组的结果。

螺旋遍历:从左上角开始,按照 向右向下向左向上 的顺序 依次 提取元素,然后再进入内部一层重复相同的步骤,直到提取完所有元素。

示例 1:

输入:array = [[1,2,3],[8,9,4],[7,6,5]]
输出:[1,2,3,4,5,6,7,8,9]

示例 2:

输入:array  = [[1,2,3,4],[12,13,14,5],[11,16,15,6],[10,9,8,7]]
输出:[1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16]

限制:

  • 0 <= array.length <= 100
  • 0 <= array[i].length <= 100

原理

  1. 初始化边界:

    • north, south, east, west 分别表示二维数组的上、下、右、左边界。开始时,north = 0, south = array.length - 1, east = array[0].length - 1, west = 0,即全覆盖数组的外边界。
  2. 螺旋遍历:

    • 螺旋遍历从左上角(north, west)开始,顺时针按“右、下、左、上”的顺序遍历二维数组。

    • 第一步: 从左到右遍历,即遍历当前的北边界行。

    • 第二步: 从上到下遍历,即遍历当前的东边界列。

    • 第三步: 从右到左遍历,即遍历当前的南边界行。

    • 第四步: 从下到上遍历,即遍历当前的西边界列。

    每一轮螺旋遍历后,都会减少遍历区域的边界:

    • north++:上边界下移
    • west++:左边界右移
    • south--:下边界上移
    • east--:右边界左移

    这样,每完成一圈遍历,内层的元素逐步被收录,直到所有元素都被遍历完。

  3. 终止条件:

    • sum(剩余未遍历元素的数量)为 0 时,表示所有元素都已被访问完,跳出循环。
  4. 返回结果:

    • 最终,list 数组包含了螺旋遍历的所有元素,按顺序返回。

时间复杂度:

  • 假设数组的维度是 m x n,遍历所有元素的时间复杂度是 O(m * n),因为每个元素都只会被访问一次。

空间复杂度:

  • 空间复杂度为 O(m * n),因为需要一个大小为 m * n 的数组来存储遍历的结果。

代码 

class Solution {public int[] spiralArray(int[][] array) {// 获取二维数组的行数,即南边界int south = array.length - 1;  // 如果数组没有行,直接返回空数组if (south == -1) {return new int[0];}// 获取二维数组的列数,即东边界int east = array[0].length - 1;  // 如果数组没有列,直接返回空数组if (east == -1) {return new int[0];}// 计算二维数组中的总元素个数,决定返回数组的大小int sum = (south + 1) * (east + 1);  // 初始化边界int north = 0, west = 0;// 创建存储结果的数组int[] list = new int[sum];// count 用来记录当前插入的位置int count = 0;// 开始螺旋遍历while (sum > 0) {// 从左到右遍历(沿着北边界)for (int i = west; i <= east; i++) {list[count++] = array[north][i];  // 将当前元素加入结果数组sum--;  // 每遍历一个元素,减去一个}// 如果所有元素已经遍历完,跳出循环if (sum == 0) {break;}// 从上到下遍历(沿着东边界)for (int i = north + 1; i <= south; i++) {list[count++] = array[i][east];  // 将当前元素加入结果数组sum--;  // 每遍历一个元素,减去一个}// 如果所有元素已经遍历完,跳出循环if (sum == 0) {break;}// 从右到左遍历(沿着南边界)for (int i = east - 1; i >= west; i--) {list[count++] = array[south][i];  // 将当前元素加入结果数组sum--;  // 每遍历一个元素,减去一个}// 如果所有元素已经遍历完,跳出循环if (sum == 0) {break;}// 从下到上遍历(沿着西边界)for (int i = south - 1; i > north; i--) {list[count++] = array[i][west];  // 将当前元素加入结果数组sum--;  // 每遍历一个元素,减去一个}// 更新边界,进入内层north++;west++;south--;east--;}// 返回螺旋遍历的结果return list;}
}

 54.螺旋矩阵

给你一个 m 行 n 列的矩阵 matrix ,请按照 顺时针螺旋顺序 ,返回矩阵中的所有元素。

示例 1:

输入:matrix = [[1,2,3],[4,5,6],[7,8,9]]
输出:[1,2,3,6,9,8,7,4,5]

示例 2:

输入:matrix = [[1,2,3,4],[5,6,7,8],[9,10,11,12]]
输出:[1,2,3,4,8,12,11,10,9,5,6,7]

提示:

  • m == matrix.length
  • n == matrix[i].length
  • 1 <= m, n <= 10
  • -100 <= matrix[i][j] <= 100

原理

  1. 初始化边界:

    • north, south, east, west 分别表示二维数组的上、下、右、左边界。开始时,north = 0, south = matrix.length - 1, east = matrix[0].length - 1, west = 0,即全覆盖矩阵的外边界。
  2. 螺旋遍历:

    • 螺旋遍历从左上角(north, west)开始,顺时针按“右、下、左、上”的顺序遍历二维矩阵中的元素。

    • 第一步: 从左到右遍历,即遍历当前的北边界行。

    • 第二步: 从上到下遍历,即遍历当前的东边界列。

    • 第三步: 从右到左遍历,即遍历当前的南边界行。

    • 第四步: 从下到上遍历,即遍历当前的西边界列。

    每完成一圈螺旋遍历后,都会更新边界:

    • north++:上边界下移
    • west++:左边界右移
    • south--:下边界上移
    • east--:右边界左移

    这样,每完成一圈遍历,内层的元素逐步被收录,直到所有元素都被遍历完。

  3. 终止条件:

    • sum(剩余未遍历元素的数量)为 0 时,表示所有元素都已被访问完,跳出循环。
  4. 返回结果:

    • 最终,list 列表包含了矩阵的螺旋顺序遍历结果,按顺序返回。

时间复杂度:

  • 假设矩阵的维度是 m x n,遍历所有元素的时间复杂度是 O(m * n),因为每个元素只会被访问一次。

空间复杂度:

  • 空间复杂度为 O(m * n),因为需要一个大小为 m * n 的列表来存储遍历的结果。

代码 

class Solution {public List<Integer> spiralOrder(int[][] matrix) {// 获取矩阵的南边界(即最后一行的索引)int south = matrix.length - 1;// 获取矩阵的东边界(即最后一列的索引)int east = matrix[0].length - 1;// 计算矩阵中元素的总数int sum = (south + 1) * (east + 1);// 初始化边界,north 是上边界,west 是左边界int north = 0, west = 0;// 用一个列表来存储螺旋遍历的结果List<Integer> list = new ArrayList<>(south * east);// 螺旋遍历开始,直到所有元素都被访问while (sum > 0) {// 1. 从左向右遍历(沿着北边界)for (int i = west; i <= east; i++) {list.add(matrix[north][i]);  // 将当前元素加入结果列表sum--;  // 减少未访问的元素数目}// 如果所有元素已经遍历完,退出循环if (sum == 0) {break;}// 2. 从上到下遍历(沿着东边界)for (int i = north + 1; i <= south; i++) {list.add(matrix[i][east]);  // 将当前元素加入结果列表sum--;  // 减少未访问的元素数目}// 如果所有元素已经遍历完,退出循环if (sum == 0) {break;}// 3. 从右向左遍历(沿着南边界)for (int i = east - 1; i >= west; i--) {list.add(matrix[south][i]);  // 将当前元素加入结果列表sum--;  // 减少未访问的元素数目}// 如果所有元素已经遍历完,退出循环if (sum == 0) {break;}// 4. 从下到上遍历(沿着西边界)for (int i = south - 1; i > north; i--) {list.add(matrix[i][west]);  // 将当前元素加入结果列表sum--;  // 减少未访问的元素数目}// 更新边界,进入内层north++;  // 上边界下移west++;   // 左边界右移south--;  // 下边界上移east--;   // 右边界左移}// 返回螺旋遍历的结果return list;}
}

59.螺旋矩阵II

给你一个正整数 n ,生成一个包含 1 到 n2 所有元素,且元素按顺时针顺序螺旋排列的 n x n 正方形矩阵 matrix 。

示例 1:

输入:n = 3
输出:[[1,2,3],[8,9,4],[7,6,5]]

示例 2:

输入:n = 1
输出:[[1]]

提示:

  • 1 <= n <= 20

原理

  1. 初始化:

    • 我们首先创建一个大小为 n x n 的矩阵 ans
    • sum 变量记录当前要填充的数字,从 1 开始递增。
    • 使用 rowcolumn 变量来指示当前的矩阵填充位置。
  2. 螺旋顺序填充:

    • 每一圈的填充包括四个方向:向右、向下、向左、向上,依次填充矩阵的边界。
    • 向右: 从当前的 (row, column) 开始,向右填充 n 个位置。
    • 向下: 在完成右边的填充后,向下填充 n-1 个位置。
    • 向左: 完成下边的填充后,向左填充 n-2 个位置。
    • 向上: 完成左边的填充后,向上填充 n-3 个位置。

    每次一圈填充后,更新边界,将 n 减少 2,以保证下一圈填充在内层矩阵中进行。

  3. 边界收缩:

    • 每次完成一圈的填充后,column++ 将列向右移动,准备进行下次填充。
    • 同时 n -= 2,表示当前有效矩阵的边长减少 2,从而让填充继续进行到内层。
  4. 终止条件:

    • n <= 0 时,表示已经填充完所有的元素,退出循环并返回结果矩阵。

时间复杂度:

  • 假设矩阵的大小为 n x n,所有元素都需要被遍历一次并填充。因此时间复杂度为 O(n²)

空间复杂度:

  • 需要一个 n x n 的矩阵来存储结果,因此空间复杂度为 O(n²)

代码

class Solution {public int[][] generateMatrix(int n) {// 初始化一个 n x n 的矩阵int[][] ans = new int[n][n];// sum 用来记录当前填充的数字,初始为 1int sum = 1;// row 和 column 用来记录当前要填充的位置的行和列int row = 0, column = 0;// 只要 n > 0,就继续填充while (n > 0) {// 1. 从左到右填充当前的北边界for (int i = 0; i < n; i++) {ans[row][column++] = sum++;  // 填充当前元素,并将列加 1}// 向左回退一列,因为 column++ 会使列超出范围column -= 1;// 2. 从上到下填充当前的东边界for (int i = 0; i < n - 1; i++) {ans[++row][column] = sum++;  // 填充当前元素,并将行加 1}// 3. 从右到左填充当前的南边界for (int i = n - 2; i >= 0; i--) {ans[row][--column] = sum++;  // 填充当前元素,并将列减 1}// 4. 从下到上填充当前的西边界for (int i = n - 2; i > 0; i--) {ans[--row][column] = sum++;  // 填充当前元素,并将行减 1}// 回到上边界的下一列,并且减少矩阵的规模(边界收缩)column++;n -= 2;  // 每一圈填充完成后,矩阵的有效边长减少 2}// 返回生成的矩阵return ans;}
}

相关文章:

leetcode--螺旋矩阵

LCR 146.螺旋遍历二维数组 给定一个二维数组 array&#xff0c;请返回「螺旋遍历」该数组的结果。 螺旋遍历&#xff1a;从左上角开始&#xff0c;按照 向右、向下、向左、向上 的顺序 依次 提取元素&#xff0c;然后再进入内部一层重复相同的步骤&#xff0c;直到提取完所有元…...

JavaScript(JS)的对象

目录 1.array 数组对象 2.String 字符串对象 3.JSON 对象&#xff08;数据载体&#xff0c;进行数据传输&#xff09; 4.BOM 浏览器对象 5.DOM 文档对象&#xff08;了解&#xff09; 1.array 数组对象 定义方式1&#xff1a;var 变量名 new Array(元素列表); 定义方式…...

基于BM1684的AI边缘服务器-模型转换,大模型一体机

介绍 我们属于SoC模式&#xff0c;即我们在x86主机上基于tpu-nntc和libsophon完成模型的编译量化与程序的交叉编译&#xff0c;部署时将编译好的程序拷贝至SoC平台&#xff08;1684开发板/SE微服务器/SM模组&#xff09;中执行。 注&#xff1a;以下都是在Ubuntu20.04系统上操…...

git推送多个仓库

在 Git 中&#xff0c;可以通过添加多个远程仓库来实现一次 git push 推送到多个仓库&#xff0c;比如同时推送到 Gitee 和 GitHub。以下是详细的设置步骤&#xff1a; 1. 添加多个远程仓库 假设你的项目已经有一个远程仓库&#xff08;例如 GitHub&#xff09;&#xff0c;你…...

Matlab mex- setup报错—错误使用 mex,未检测到支持的编译器...

错误日志&#xff1a; 在使用mex编译时报错提示&#xff1a;错误使用 mex&#xff0c;未检测到支持的编译器。您可以安装免费提供的 MinGW-w64 C/C 编译器&#xff1b;请参阅安装 MinGW-w64 编译器。有关更多选项&#xff0c;请访问https://www.mathworks.com/support/compile…...

PostgreSQL认证培训需要什么条件

PostgreSQL认证培训通常没有严格的前置条件&#xff0c;但以下几点可以帮助你更好地准备和通过认证考试&#xff1a; 1、基础知识&#xff1a;具备基本的数据库知识和经验&#xff0c;特别是对SQL有一定的了解。如果你Oracle、MySQL等基础知识&#xff0c;对对你学习PostgreSQ…...

Oracle—系统包使用

文章目录 系统包dbms_redefinition 系统包 dbms_redefinition 功能介绍&#xff1a;该包体可以实现将Oracle库下的表在线改为分区结构或者重新定义&#xff1b; 说明&#xff1a;在检查表是否可以重定义和开始重定义的过程中&#xff0c;按照表是否存在主键&#xff0c;参数 o…...

【排序用法】.NET开源 ORM 框架 SqlSugar 系列

&#x1f4a5; .NET开源 ORM 框架 SqlSugar 系列 &#x1f389;&#x1f389;&#x1f389; 【开篇】.NET开源 ORM 框架 SqlSugar 系列【入门必看】.NET开源 ORM 框架 SqlSugar 系列【实体配置】.NET开源 ORM 框架 SqlSugar 系列【Db First】.NET开源 ORM 框架 SqlSugar 系列…...

【SpringBoot】整合篇

1、log4j2 第一步&#xff0c;导入依赖 <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <exclusions><!-- 去掉springboot默认配置 --> <exclusion> <…...

写入json和读取json文件

/// <summary> ///写入文件 /// </summary> /// <param name"Stns"></param> /// <returns></returns> public ActionResult WriteJsonFile(string Stns) { strin…...

Vuex的理解及使用场景

Vuex 是 Vue.js 应用中一个专门为状态管理而设计的库&#xff0c;它基于 Fluts 和 Redux 的模式。Vuex 提供了一种集中式存储管理所有组件的状态&#xff0c;并以相应的规则保证状态以一种可预测的方式发生变化。以下是 Vuex 的理解及使用场景&#xff1a; Vuex 的理解 核心概…...

PostGis学习笔记

– 文本方式查看几何数据 SELECT ST_AsText(geom)FROM nyc_streets WHERE name ‘Avenue O’; – 计算紧邻的街区 SELECT name,ST_GeometryType(geom) FROM nyc_streets WHERE ST_DWithin( geom,ST_GeomFromText(‘LINESTRING(586782 4504202,586864 4504216)’,26918),0.1); …...

Qt 窗口类型、窗口标志和窗口属性

一、窗口类型 Qt 窗口标志枚举类型用于指定小部件的各种窗口系统属性。其中一些标志取决于底层窗口管理器是否支持它们。以下是窗口类型: Qt::QWidget:这是 QWidget 的默认类型。如果它们有父级,这种类型的部件是子部件,如果没有父控件,则为独立窗口。Qt::Window:通常具…...

相机学习笔记——工业相机的基本参数

0、相机分类 图像颜色不同可以分为黑白相机和彩色相机&#xff1a;相同分辨率下&#xff0c;黑白工业相机相比彩色工业相机精度更高&#xff0c;检测图像边缘时&#xff0c;黑白工业相机成像效果更好。 芯片类型不同可以分为CCD相机和CMOS相机&#xff1a;CCD工业相机具有体积小…...

MATLAB - ROS2 ros2genmsg 生成自定义消息(msg/srv...)

系列文章目录 前言 语法 ros2genmsg(folderpath)ros2genmsg(folderpath,Name=Value) 一、说明 ros2genmsg(folderpath) 通过读取指定文件夹路径下的 ROS 2 自定义信息和服务定义来生成 ROS 2 自定义信息。函数文件夹必须包含一个或多个 ROS 2 软件包。这些软件包包含 .msg 文…...

【Git 操作】-- 将 fork master 分支的最新commit更新到自己的仓库

目录 1.举例 2. 配置上游仓库&#xff08;Upstream&#xff09; 3. 获取上游仓库的更新 4. 切换到你自己的 master 分支 5. 合并上游仓库的 master 分支 6. 解决冲突&#xff08;如果有的话&#xff09; 7. 推送更新到你自己的 GitHub 仓库 1.举例 当我们从 github 的 h…...

[高等数学学习记录] 泰勒公式

1 知识点 1.1 要求 为简化计算, 通常用多项式近似表达复杂函数: 设函数 f ( x ) f(x) f(x) 在含有 x 0 x_0 x0​ 的开区间内具有 ( n 1 ) (n1) (n1) 阶导数, 试找出一个关于 ( x − x 0 ) (x-x_0) (x−x0​) 的 n n n 次多项式 p n ( x ) p_n(x) pn​(x) 近似表达 f…...

我的创作纪念日—128天的坚持|分享|成长

&#x1f4ab;《博主介绍》&#xff1a;✨又是一天没白过&#xff0c;我是奈斯&#xff0c;DBA一名✨ &#x1f4ab;《擅长领域》&#xff1a;✌️擅长Oracle、MySQL、SQLserver、阿里云AnalyticDB for MySQL(分布式数据仓库)、Linux&#xff0c;也在扩展大数据方向的知识面✌️…...

万字长文解读深度学习——多模态模型BLIP2

&#x1f33a;历史文章列表&#x1f33a; 深度学习——优化算法、激活函数、归一化、正则化 深度学习——权重初始化、评估指标、梯度消失和梯度爆炸 深度学习——前向传播与反向传播、神经网络&#xff08;前馈神经网络与反馈神经网络&#xff09;、常见算法概要汇总 万字长…...

selinux与防火墙

selinux 什么是selinux SELinux 是 Security-Enhanced Linux 的缩写&#xff0c;意思是安全强化的 linux 。 SELinux 主要由美国国家安全局&#xff08; NSA &#xff09;开发&#xff0c;当初开发的目的是为了避免资源的误用。 系统资源都是通过程序进行访问的&#xff0…...

基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真

目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销&#xff0c;平衡网络负载&#xff0c;延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...

K8S认证|CKS题库+答案| 11. AppArmor

目录 11. AppArmor 免费获取并激活 CKA_v1.31_模拟系统 题目 开始操作&#xff1a; 1&#xff09;、切换集群 2&#xff09;、切换节点 3&#xff09;、切换到 apparmor 的目录 4&#xff09;、执行 apparmor 策略模块 5&#xff09;、修改 pod 文件 6&#xff09;、…...

日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする

日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする 1、前言(1)情况说明(2)工程师的信仰2、知识点(1) にする1,接续:名词+にする2,接续:疑问词+にする3,(A)は(B)にする。(2)復習:(1)复习句子(2)ために & ように(3)そう(4)にする3、…...

UE5 学习系列(三)创建和移动物体

这篇博客是该系列的第三篇&#xff0c;是在之前两篇博客的基础上展开&#xff0c;主要介绍如何在操作界面中创建和拖动物体&#xff0c;这篇博客跟随的视频链接如下&#xff1a; B 站视频&#xff1a;s03-创建和移动物体 如果你不打算开之前的博客并且对UE5 比较熟的话按照以…...

聊聊 Pulsar:Producer 源码解析

一、前言 Apache Pulsar 是一个企业级的开源分布式消息传递平台&#xff0c;以其高性能、可扩展性和存储计算分离架构在消息队列和流处理领域独树一帜。在 Pulsar 的核心架构中&#xff0c;Producer&#xff08;生产者&#xff09; 是连接客户端应用与消息队列的第一步。生产者…...

Opencv中的addweighted函数

一.addweighted函数作用 addweighted&#xff08;&#xff09;是OpenCV库中用于图像处理的函数&#xff0c;主要功能是将两个输入图像&#xff08;尺寸和类型相同&#xff09;按照指定的权重进行加权叠加&#xff08;图像融合&#xff09;&#xff0c;并添加一个标量值&#x…...

ios苹果系统,js 滑动屏幕、锚定无效

现象&#xff1a;window.addEventListener监听touch无效&#xff0c;划不动屏幕&#xff0c;但是代码逻辑都有执行到。 scrollIntoView也无效。 原因&#xff1a;这是因为 iOS 的触摸事件处理机制和 touch-action: none 的设置有关。ios有太多得交互动作&#xff0c;从而会影响…...

微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据

微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据 Power Query 具有大量专门帮助您清理和准备数据以供分析的功能。 您将了解如何简化复杂模型、更改数据类型、重命名对象和透视数据。 您还将了解如何分析列&#xff0c;以便知晓哪些列包含有价值的数据&#xff0c;…...

MySQL JOIN 表过多的优化思路

当 MySQL 查询涉及大量表 JOIN 时&#xff0c;性能会显著下降。以下是优化思路和简易实现方法&#xff1a; 一、核心优化思路 减少 JOIN 数量 数据冗余&#xff1a;添加必要的冗余字段&#xff08;如订单表直接存储用户名&#xff09;合并表&#xff1a;将频繁关联的小表合并成…...

快刀集(1): 一刀斩断视频片头广告

一刀流&#xff1a;用一个简单脚本&#xff0c;秒杀视频片头广告&#xff0c;还你清爽观影体验。 1. 引子 作为一个爱生活、爱学习、爱收藏高清资源的老码农&#xff0c;平时写代码之余看看电影、补补片&#xff0c;是再正常不过的事。 电影嘛&#xff0c;要沉浸&#xff0c;…...