LeetCode54 螺旋矩阵,题目不重要,重要的是这个技巧

2024-03-21 14:32

本文主要是介绍LeetCode54 螺旋矩阵,题目不重要,重要的是这个技巧,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文始发于个人公众号:TechFlow,原创不易,求个关注


今天是LeetCode专题的第32篇文章,我们一起看的是LeetCode的第54题——Spiral Matrix。

首先解释一下题意,这个Spiral是螺旋的意思,据说英文版的漫画里,把鸣人的螺旋丸就翻译成Spiral Sphere…

走远了,回归正传。通过螺旋丸我们都知道螺旋形是什么意思,所以所谓的螺旋矩阵,就是按照螺旋形的顺序来遍历一个数组,或者说矩阵。我们可以看下下图:

箭头标注的顺序就是螺旋的顺序,这道题让我们求的就是按照螺旋的顺序遍历之后的结果。

背景

这道题的题意非常简单,我想大家肯定都能看明白,但是真的要上手去做会发现还是蛮困难的。主要的难点就是我们遍历的顺序一直在变化,并且变化的速度也是在变化的。虽然从某种程度上来说,这些变化都是有迹可循的,但是即便如此,把这些规律抽象出来写成简单易懂没有bug的代码也并不是一件容易的事情。

如果我没有记错的话,当年我大二的时候学校里的acm校赛有一题就是这个,一模一样的原题。虽然非常简单,每个人都知道怎么做,但是最后的通过率并不高。

这并不完全是出题人的恶意,其实这类问题在acm的比赛当中还是很常见的。经常会有一些题目很清晰明确,你只需要照着实现就可以了的题目。考察的就是选手的抽象和编码能力,虽然题意不难,但是在比赛高压的场景下想要快速写出一个几百行包含一系列复杂逻辑的程序并且没有bug,还是非常困难的一件事。由于这类题目的关键就是让我们模拟出来题目的意思,所以也被称为模拟题。

想要比较顺手地写出这道题,需要一个很常用的技巧或者说惯例,这也是这篇文章的关键。

方向数组

在许多问题当中我们经常会遇到这样一个问题,比如我们需要遍历一个迷宫,需要沿着现实世界当中上下左右或者是东南西北的方向进行移动。在移动的过程当中自然就涉及各种各样的方向,于是我们需要用代码来表示方向。

比如我们画一个小人,它所在的坐标是(x, y),它有东南西北四个方向可以选。我们假设他每次移动一个单位的距离,我们很容易就得出它往各个方向移动之后的新坐标。

根据数学上向量的定义,我们可以写出这四个方向的方向单位向量:[0, 1], [0, -1], [1, 0], [-1, 0]。

我们把这些向量存放进一个常量数组当中,就得到了方向数组。当我们需要遍历所有方向的时候,我们只需要遍历这个数组即可。

不仅如此,如果我们对方向的遍历顺序有要求,它也完全可以实现。比如在这题当中,我们可以发现,螺旋遍历每一次改变顺序其实都是向左转了90度

原本朝东的旋转之后变成了朝南,朝南的变成了朝西,朝西的成了朝北,而朝北的成了朝东。那么我们只需要把方向按照东南西北的顺序摆放,我们每次把方向数组的下标增加一位并对4取模,就得到了下一个方向。

这个技巧并不难理解,但是可以大大简化我们的代码。

解答

理解了方向数组之后剩下的就容易多了,我们观察一下上面螺旋遍历的过程,每一次改变方向遍历的长度虽然不同,但是每一次改变的原因是一致的,就是这个方向上已经遍历到头了,所以我们需要变更方向。明白了这点其实就很容易了,我们只需要维护每个方向上的终点,每次到终点则进行变向。由于矩阵当中元素的数量是固定的,我们遍历的次数也就知道了,所以只要把变更方向的事情处理好,这道题也就解决了。

我们来看下代码:

class Solution:def spiralOrder(self, matrix: List[List[int]]) -> List[int]:# 定义方向数组fx = [[0, 1], [1, 0], [0, -1], [-1, 0]]n = len(matrix)if n == 0:return []m = len(matrix[0])ret = []# 定义边界数组# 边界数组和旋转的顺序也是对应的# 第一次旋转之后上边界+1,所以第0位是上边界# 第二次旋转之后,右边界-1# 以此类推condition = [0, m-1, n-1, 0]x, y, dt = 0, 0, 0for i in range(n * m):ret.append(matrix[x][y])x_, y_ = x + fx[dt][0], y + fx[dt][1]# 如果已经越过边界了,则需要转向if x_ < condition[0] or x_ > condition[2] or y_ < condition[3] or y_ > condition[1]:# 更新边界if dt in (1, 2):condition[dt] -= 1else:condition[dt] += 1# 转向,并且重新移动dt = (dt + 1) % 4x_, y_ = x+fx[dt][0], y+fx[dt][1]# print(x_, y_)x, y = x_, y_return ret

我们观察一下代码,还有一个我们刚才没有提到的细节。我们在移动的时候,并不是直接在原变量上进行变更,因为如果一旦移动超界或者触发其他非法的情况,那么我们就无法回滚了。所以我们会创建新的x和y的变量来表示移动之后的位置,即使移动到了非法位置,也不会影响之前的结果。这也是一个常用的技巧,在Python当中,我们在变量末尾加上下划线表示这是一个影子(克隆)变量。

总结

我个人认为今天的题目出得不错,初学者很有必要亲自动手做一下。因为在做题的过程当中我们可以很具体地学到方向数组这个很有用的解题技巧,它在搜索问题当中广泛使用,可以说是做算法题必须会的技巧之一。

可能你会觉得我们通过边界判断是否需要转向的逻辑看起来有些复杂,这并不是必须的。我们也可以使用一些其他方法来代替,比如我们可以开辟一个数组记录已经遍历过的位置来代替边界的判定,和使用边界判定的方法相比,这样做消耗的空间要更大一些,并且通过边界判定的方法更加考验思维一些,因此我个人比较推荐。当然,这题还有一些其他的方法,比如找规律什么的,不是特别巧妙,就不占用大家时间了。

今天的文章就是这些,如果觉得有所收获,请顺手点个关注或者转发吧,你们的举手之劳对我来说很重要。

在这里插入图片描述

这篇关于LeetCode54 螺旋矩阵,题目不重要,重要的是这个技巧的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/832897

相关文章

电脑提示Winmm.dll缺失怎么办? Winmm.dll文件丢失的多种修复技巧

《电脑提示Winmm.dll缺失怎么办?Winmm.dll文件丢失的多种修复技巧》有时电脑会出现无法启动程序,因为计算机中丢失winmm.dll的情况,其实,winmm.dll丢失是一个比较常见的问... 在大部分情况下出现我们运行或安装软件,游戏出现提示丢失某些DLL文件或OCX文件的原因可能是原始安装包

ubuntu16.04如何部署dify? 在Linux上安装部署Dify的技巧

《ubuntu16.04如何部署dify?在Linux上安装部署Dify的技巧》随着云计算和容器技术的快速发展,Docker已经成为现代软件开发和部署的重要工具之一,Dify作为一款优秀的云原生应用... Dify 是一个基于 docker 的工作流管理工具,旨在简化机器学习和数据科学领域的多步骤工作流。它

MySQL复合查询从基础到多表关联与高级技巧全解析

《MySQL复合查询从基础到多表关联与高级技巧全解析》本文主要讲解了在MySQL中的复合查询,下面是关于本文章所需要数据的建表语句,感兴趣的朋友跟随小编一起看看吧... 目录前言:1.基本查询回顾:1.1.查询工资高于500或岗位为MANAGER的雇员,同时还要满足他们的姓名首字母为大写的J1.2.按照部门

Python中Flask模板的使用与高级技巧详解

《Python中Flask模板的使用与高级技巧详解》在Web开发中,直接将HTML代码写在Python文件中会导致诸多问题,Flask内置了Jinja2模板引擎,完美解决了这些问题,下面我们就来看看F... 目录一、模板渲染基础1.1 为什么需要模板引擎1.2 第一个模板渲染示例1.3 模板渲染原理二、模板

Pandas利用主表更新子表指定列小技巧

《Pandas利用主表更新子表指定列小技巧》本文主要介绍了Pandas利用主表更新子表指定列小技巧,通过创建主表和子表的DataFrame对象,并使用映射字典进行数据关联和更新,实现了从主表到子表的同... 目录一、前言二、基本案例1. 创建主表数据2. 创建映射字典3. 创建子表数据4. 更新子表的 zb

快速修复一个Panic的Linux内核的技巧

《快速修复一个Panic的Linux内核的技巧》Linux系统中运行了不当的mkinitcpio操作导致内核文件不能正常工作,重启的时候,内核启动中止于Panic状态,该怎么解决这个问题呢?下面我们就... 感谢China编程(www.chinasem.cn)网友 鸢一雨音 的投稿写这篇文章是有原因的。为了配置完

Python ZIP文件操作技巧详解

《PythonZIP文件操作技巧详解》在数据处理和系统开发中,ZIP文件操作是开发者必须掌握的核心技能,Python标准库提供的zipfile模块以简洁的API和跨平台特性,成为处理ZIP文件的首选... 目录一、ZIP文件操作基础三板斧1.1 创建压缩包1.2 解压操作1.3 文件遍历与信息获取二、进阶技

Java字符串操作技巧之语法、示例与应用场景分析

《Java字符串操作技巧之语法、示例与应用场景分析》在Java算法题和日常开发中,字符串处理是必备的核心技能,本文全面梳理Java中字符串的常用操作语法,结合代码示例、应用场景和避坑指南,可快速掌握字... 目录引言1. 基础操作1.1 创建字符串1.2 获取长度1.3 访问字符2. 字符串处理2.1 子字

Java Optional的使用技巧与最佳实践

《JavaOptional的使用技巧与最佳实践》在Java中,Optional是用于优雅处理null的容器类,其核心目标是显式提醒开发者处理空值场景,避免NullPointerExce... 目录一、Optional 的核心用途二、使用技巧与最佳实践三、常见误区与反模式四、替代方案与扩展五、总结在 Java

Python通过模块化开发优化代码的技巧分享

《Python通过模块化开发优化代码的技巧分享》模块化开发就是把代码拆成一个个“零件”,该封装封装,该拆分拆分,下面小编就来和大家简单聊聊python如何用模块化开发进行代码优化吧... 目录什么是模块化开发如何拆分代码改进版:拆分成模块让模块更强大:使用 __init__.py你一定会遇到的问题模www.