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

Leetcode 螺旋矩阵

在这里插入图片描述

算法思想:

这个算法的目标是按照顺时针螺旋的顺序从矩阵中取出元素。为了做到这一点,整个思路可以分成几个关键步骤:

  1. 定义边界:首先需要定义四个边界变量:

    • left:当前左边界的索引。
    • right:当前右边界的索引。
    • top:当前上边界的索引。
    • bottom:当前下边界的索引。

    这些变量一开始会对应矩阵的外边界,然后我们会通过逐步缩小这些边界来遍历矩阵的每一层。

  2. 螺旋遍历:接下来我们会按照四个方向进行遍历:

    • 从左到右:遍历上边界的所有元素,从左到右遍历,遍历结束后将上边界向下移动一行(top++)。
    • 从上到下:遍历右边界的所有元素,从上到下遍历,遍历结束后将右边界向左移动一列(right--)。
    • 从右到左:如果还有剩余行需要遍历(即top <= bottom),遍历下边界的所有元素,从右到左遍历,遍历结束后将下边界向上移动一行(bottom--)。
    • 从下到上:如果还有剩余列需要遍历(即left <= right),遍历左边界的所有元素,从下到上遍历,遍历结束后将左边界向右移动一列(left++)。
  3. 边界条件检查:在每次从右到左或从下到上遍历之前,需要检查是否仍然有需要遍历的行或列。这是因为在某些情况下,矩阵的某些边界可能已经被其他方向的遍历覆盖了。

  4. 终止条件:当left > right或者top > bottom时,意味着已经没有元素可以遍历了,这时循环终止。

算法流程:

  • 我们从矩阵的左上角开始,按照顺时针的顺序依次遍历。
  • 每完成一圈的遍历,我们就收缩边界,继续处理内层的矩阵,直到所有元素都被处理为止。

时间复杂度分析:

  • 每次遍历矩阵中的一圈元素,而每个元素最多被访问一次,因此时间复杂度是 O(m * n),其中 m 是矩阵的行数,n 是列数。
class Solution {
    public List<Integer> spiralOrder(int[][] matrix) {
        //首先创建一个存储结果的ArrayList
        List<Integer> result = new ArrayList <>();
        
        //获得矩阵的行和列
        int rows = matrix.length;
        int cols = matrix[0].length;

        //然后定义四周墙壁
        int left = 0;
        int right = cols - 1;
        int top = 0;
        int bottom = rows - 1;

        //开始遍历
        //最后一个元素是矩阵正中心时(left = result && top = bottom),所以循环终止条件是小于等于
        while(left <= right && top <= bottom) { 
            //先从左往右遍历上方的每一行, i 从 left 开始遍历到 right, 而不是从 0 开始
            for(int i = left; i <= right; i++) {
                result.add(matrix[top][i]);
            }
            //每遍历完上方的一行,需要更新 top
            top++;

            //然后从上往下遍历最右边一列
            for(int i = top; i <= bottom; i++) {
                result.add(matrix[i][right]);
            }
            //每次遍历完最右边列,需要更新right
            right--;

            if (top <= bottom) {//检查是否还有需要遍历的行
                //然后从右往左遍历最下方的行
                for(int i = right; i >= left; i--) {
                    result.add(matrix[bottom][i]);
                }
                //每次遍历完最下行,需要更新bottom
                bottom--;
            }

            if (left <= right) {//检查是否还有需要遍历的列
                //然后下往上遍历最左方的列
                for(int i = bottom; i >= top; i--) {
                    result.add(matrix[i][left]);
                }
                //每次遍历完最左方的列,需要更新left
                left++;
            }
        }
        return result;
    }
}

为什么这一部分代码片段,需要使用2个 if 进行判断?

            if (top <= bottom) {
                // Traverse from right to left
                for (int i = right; i >= left; i--) {
                    result.add(matrix[bottom][i]);
                }
                bottom--;
            }

            if (left <= right) {
                // Traverse from bottom to top
                for (int i = bottom; i >= top; i--) {
                    result.add(matrix[i][left]);
                }
                left++;
            }

在这段代码中,两个 if 判断是用来确保在进行从右到左、从下到上的遍历时,矩阵的边界没有发生交叉。如果不进行这些判断,可能会出现重复遍历或错误遍历的情况。

让我们具体分析这两个 if 判断的作用:

1. 第一个 if 判断 (if (top <= bottom))

if (top <= bottom) {
    // Traverse from right to left
    for (int i = right; i >= left; i--) {
        result.add(matrix[bottom][i]);
    }
    bottom--;
}
  • 这个 if 判断用于检查当前上边界 top 是否仍然小于等于下边界 bottom,也就是说,检查是否还有需要遍历的行。如果 top > bottom,说明矩阵的所有行已经遍历完了,再进行从右到左的操作将是没有意义的,甚至可能导致重复遍历或越界。

  • 如果 top <= bottom,则表示当前还有未处理的行,于是可以安全地从右边界 right 开始向左遍历,并在遍历结束后将下边界 bottom 向上移动,减少一行。

2. 第二个 if 判断 (if (left <= right))

if (left <= right) {
    // Traverse from bottom to top
    for (int i = bottom; i >= top; i--) {
        result.add(matrix[i][left]);
    }
    left++;
}
  • 这个 if 判断用于检查当前左边界 left 是否仍然小于等于右边界 right,也就是说,检查是否还有需要遍历的列。如果 left > right,说明矩阵的所有列已经遍历完了,再进行从下到上的操作将是没有必要的,可能导致重复操作或者越界。

  • 如果 left <= right,则表示当前还有未处理的列,可以从下边界 bottom 向上遍历左边界的元素,并在遍历结束后将左边界 left 向右移动,减少一列。

为什么需要两个独立的 if 判断?

  1. 防止重复遍历:在螺旋遍历中,每次都会收缩边界(上、下、左、右)。当已经遍历完一层的矩阵后,边界会逐渐靠近矩阵的中心。如果没有这些 if 判断,在某些情况下,比如在最后一层矩阵中,可能会导致重复遍历某些行或列。例如,如果你在遍历完右到左的行后,紧接着遍历从下到上的列,而此时没有检查边界是否已经重叠,可能会再次遍历已经处理过的行或列。

  2. 防止越界:螺旋遍历会不断缩小边界,如果不做这些检查,当边界重叠或交叉时,程序可能会试图访问无效的数组索引,导致数组越界异常。通过在每次遍历前检查边界的有效性,可以确保访问的数组索引始终在合法范围内。

举例:

假设你有一个 3x3 的矩阵:

1  2  3
4  5  6
7  8  9

按顺时针遍历的顺序是:

  • 从左到右遍历第一行 [1, 2, 3]
  • 从上到下遍历第三列 [6, 9]
  • 从右到左遍历第三行 [8, 7]
  • 从下到上遍历第一列 [4]
  • 最后剩下中心元素 [5]

在这过程中,随着边界逐渐收缩,如果没有 if 判断,在遍历完成第三行时,可能会再次从右到左遍历已处理过的行,或者尝试遍历不存在的行。两个 if 判断就是为了避免这种情况发生。

总结:

  • if (top <= bottom)if (left <= right) 的作用是确保遍历时矩阵的边界没有交叉,从而避免重复访问或越界访问。
  • 它们是为了保证代码在任意大小和形状的矩阵上都能正确运行,特别是在螺旋遍历的过程中边界不断收缩的情况下。

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

相关文章:

  • arm 汇编技巧
  • Spring——容器:IoC
  • 数据库管理-第260期 业务向前,数据库向后(20241111)
  • uniapp vue里按钮上的文字,换行的方法,用rich-text
  • 计算机毕业设计Python+大模型斗鱼直播可视化 直播预测 直播爬虫 直播数据分析 直播大数据 大数据毕业设计 机器学习 深度学习
  • 【C语言刷力扣】66.加一
  • 关于idea编辑xml文件卡死
  • 选择租用徐州服务器机柜的作用有哪些?
  • 统信服务器操作系统【开机自启动】配置方法
  • 关于前端vue3+element-plus项目正常安装运行时未报错,但是前端界面老是空白问题及解决方案(其他使用nodejs的框架同理)
  • Python记录
  • Unity Debug时出现请选择unity实例
  • 探索C语言与Linux编程:获取当前用户ID与进程ID
  • QT中的消息机制(事件机制)总结
  • 接口调用方式2
  • C语言 | Leetcode C语言题解之第434题字符串中的单词数
  • uniapp组件封装和父子组件间通讯的介绍和详细案例
  • UE学习篇ContentExample解读------Blueprint_Communication-上
  • 海康HIK IN客户端使用帮助说明
  • 【工具】语音朗读PDF的免费工具
  • 探索未来科技的深邃海洋:IT领域的波澜壮阔
  • 目标检测DOTA数据集
  • 助力智能化农田作物除草,基于YOLOv10全系列【n/s/m/b/l/x】参数模型开发构建农田作物场景下玉米苗、杂草检测识别分析系统
  • 算法:69.x的平方根
  • 力扣每日一题 字符串中最多数目的子序列 贪心 字符串 前缀和
  • JavaWeb--纯小白笔记06:使用Idea创建Web项目,Servlet生命周期,注解,中文乱码解决