螺旋矩阵的算法刷题

螺旋矩阵的算法刷题

本文主要涉及螺旋矩阵的算法
包括三个题目分别是

  1. 59. 螺旋矩阵 II
  2. 54. 螺旋矩阵 中等
  3. LCR 146. 螺旋遍历二维数组

文章目录

  • 螺旋矩阵的算法刷题
    • 一 、螺旋矩阵简单
      • 1.1 实现一(我认为这个方法更巧妙!!)
      • 1.2 实现二(经典方法--更加直观)
    • 二、螺旋矩阵 中等
      • 2.1 实现一(经典)
      • 2.2 实现二(精妙!!)
    • 三、螺旋遍历矩阵 简单

一 、螺旋矩阵简单

59. 螺旋矩阵 II

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

在这里插入图片描述

1.1 实现一(我认为这个方法更巧妙!!)

思路
采用了类似螺旋走位的方式进行填充,具体实现如下:

  1. 首先,定义一个n x n的二维数组res来存储结果。
  2. 初始化方向向量dx和dy,初始值分别为0和1,表示向右移动。
  3. 初始化坐标x和y,初始值都为0,表示从矩阵的左上角开始填充。
  4. 使用一个循环,从1到n * n遍历每个要填充的数字。
  5. 在每一步中,将当前位置的值设置为当前遍历的数字i,并根据当前方向向量(dx, dy)进行移动到下一个位置。
  6. 判断下一个位置是否已经被填充过,如果是,则表示已经到达边界,需要调转方向向量以继续填充;如果不是,则继续沿当前方向向量移动。
  7. 返回填充完成的二维数组res作为结果。

这样的算法可以确保填充的数字形成了一个螺旋状的矩阵。

C++代码实现

class Solution {
public:
    vector<vector<int>> generateMatrix(int n) {
        vector<vector<int>> matrix(n,vector<int>(n,0));
        //代表方向的向量变量 01--右 10--下 -1 0--左 0 -1 --上
        int dx=0,dy=1;
        //x,y表示当前坐标
        int x=0,y=0;
        //使用for进行循环,i表示当前值
        for(int i =1; i<=n*n;i++)
        {
            matrix[x][y]=i;
            //**关键部分**
            if(matrix[(x+dx+n)%n][(y+dy+n)%n]!=0)//判断是否已经填充,如果已经填充则向量需要换方向
            {
                int temp = dy;
                dy=-dx;
                dx=temp;
            }
            x+=dx;
            y+=dy;
        }
        return matrix;    
    }
};

重要的是下面这段代码

 if(matrix[(x+dx+n)%n][(y+dy+n)%n]!=0)//判断是否已经填充,如果已经填充则向量需要换方向
            {
                int temp = dy;
                dy=-dx;
                dx=temp;
            }

这段代码是在判断当前位置的下一个位置是否已经被填充过。如果下一个位置已经被填充过,说明当前位置已经是当前层的最后一个位置,需要改变填充方向。

逐行解释这段代码:

  1. (x + dx + n) % n:这里计算了下一个位置的横坐标。由于是螺旋填充,因此下一个位置可能超出边界。使用 % n 可以将超出边界的情况转化为在边界内的情况。例如,当 x + dx 大于等于 n 时,超出了右边界,通过 % n 可以将其映射到左边界,从而形成循环填充。
  2. 同理,(y + dy + n) % n 计算了下一个位置的纵坐标。
  3. res[(x + dx + n) % n][(y + dy + n) % n] != 0:这个条件判断当前位置的下一个位置是否已经被填充过。如果下一个位置不等于0,说明已经被填充过。
  4. 如果下一个位置已经被填充过,则需要改变填充方向。通过交换 dxdy 实现了向下转向的功能。
  5. int tem = dy; dy = -dx; dx = tem;:这里使用了一个临时变量 tem,先将 dy 的值保存下来,然后将 dy 设置为 -dxdx 设置为 tem,实现了方向的转换。

在这里插入图片描述

总的来说,这段代码的作用是在当前位置的下一个位置已经被填充过时,改变填充方向,从而继续填充螺旋矩阵。

1.2 实现二(经典方法–更加直观)

一个常见的方法是按层次遍历,逐层填充螺旋矩阵。具体步骤如下:

  1. 初始化一个n x n的二维数组res,用于存储结果。
  2. 定义四个变量top、bottom、left、right,分别表示当前填充层的上下左右边界。
  3. 初始化这些边界值:top=0,bottom=n-1,left=0,right=n-1。
  4. 初始化一个变量num,表示当前要填充的数字,初始值为1。
  5. 在一个循环中,依次填充每一层的数字,直到所有位置都被填充为止。
    • 从左到右,将当前层的最上一行从left到right填充为num开始递增的数字,同时更新top的值使其向下移动一行。
    • 从上到下,将当前层的最右一列从top到bottom填充为num开始递增的数字,同时更新right的值使其向左移动一列。
    • 从右到左,将当前层的最下一行从right到left填充为num开始递增的数字,同时更新bottom的值使其向上移动一行。
    • 从下到上,将当前层的最左一列从bottom到top填充为num开始递增的数字,同时更新left的值使其向右移动一列。
    • 每填充完一行或一列后,将num递增。
  6. 当top > bottom 或 left > right时,表示所有位置都被填充完毕,退出循环。
  7. 返回填充完成的二维数组res作为结果。

这样的实现方法更直观清晰,容易理解和实现,并且效率也很高。

class Solution {
public:
    vector<vector<int>> generateMatrix(int n) {
        vector<vector<int>> res(n, vector<int>(n, 0)); // 初始化结果矩阵为全零
        int top = 0, bottom = n - 1, left = 0, right = n - 1; // 初始化边界
        int num = 1; // 当前要填充的数字,初始为1
        
        while (true) {
            // 从左到右,填充最上一行
            for (int i = left; i <= right; ++i) {
                res[top][i] = num++;
            }
            if (++top > bottom) break; // 更新上边界,若超过下边界则退出
            
            // 从上到下,填充最右一列
            for (int i = top; i <= bottom; ++i) {
                res[i][right] = num++;
            }
            if (--right < left) break; // 更新右边界,若超过左边界则退出
            
            // 从右到左,填充最下一行
            for (int i = right; i >= left; --i) {
                res[bottom][i] = num++;
            }
            if (--bottom < top) break; // 更新下边界,若超过上边界则退出
            
            // 从下到上,填充最左一列
            for (int i = bottom; i >= top; --i) {
                res[i][left] = num++;
            }
            if (++left > right) break; // 更新左边界,若超过右边界则退出
        }
        
        return res;
    }
};

二、螺旋矩阵 中等

54. 螺旋矩阵 中等

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

在这里插入图片描述

2.1 实现一(经典)

class Solution {
public:
    vector<int> spiralOrder(vector<vector<int>>& matrix) {
        vector<int> ans;
        if (matrix.empty()) return ans;

        int m = matrix.size(); // 行数
        int n = matrix[0].size(); // 列数
        int top = 0, bottom = m - 1, left = 0, right = n - 1;

        while (top <= bottom && left <= right) {
            // 从左到右遍历上边界
            for (int j = left; j <= right; ++j) {
                ans.push_back(matrix[top][j]);
            }
            ++top;

            // 从上到下遍历右边界
            for (int i = top; i <= bottom; ++i) {
                ans.push_back(matrix[i][right]);
            }
            --right;

            // 从右到左遍历下边界
            if (top <= bottom) { // 防止重复访问上一次遍历的元素
                for (int j = right; j >= left; --j) {
                    ans.push_back(matrix[bottom][j]);
                }
                --bottom;
            }

            // 从下到上遍历左边界
            if (left <= right) { // 防止重复访问上一次遍历的元素
                for (int i = bottom; i >= top; --i) {
                    ans.push_back(matrix[i][left]);
                }
                ++left;
            }
        }

        return ans;
    }
};

按照螺旋顺序遍历二维矩阵。下面是对代码的详细解释:

  1. 首先,定义一个空的整数向量 ans 用于存储遍历结果。
  2. 如果输入的二维矩阵 matrix 为空,直接返回空的结果向量 ans
  3. 获取矩阵的行数 m 和列数 n,以及初始化四个边界变量 topbottomleftright
  4. 在一个循环中,不断遍历矩阵的边界元素,直到边界收缩到无效为止。
    • 从左到右遍历上边界,将上边界的元素依次加入结果向量 ans 中,并将 top 上移一行。
    • 从上到下遍历右边界,将右边界的元素依次加入结果向量 ans 中,并将 right 右移一列。
    • 从右到左遍历下边界,将下边界的元素依次加入结果向量 ans 中,并将 bottom 下移一行(注意要判断 top <= bottom,防止重复遍历)。
    • 从下到上遍历左边界,将左边界的元素依次加入结果向量 ans 中,并将 left 左移一列(注意要判断 left <= right,防止重复遍历)。
  5. 循环结束后,返回结果向量 ans

利用边界变量控制了螺旋顺序的遍历过程,实现了按照螺旋顺序遍历二维矩阵的功能。

2.2 实现二(精妙!!)

下面是一段Python代码:

def spiralOrder(self, matrix: List[List[int]]) -> List[int]:
        res = []
        while matrix:
            # 削头(第一层)
            res += matrix.pop(0)
            # 将剩下的逆时针转九十度,等待下次被削
            matrix = list(zip(*matrix))[::-1]
        return res

采用了不断“削头”的方式来实现。具体的步骤如下:

  1. 创建一个空列表 res 用于存储螺旋顺序遍历的结果。
  2. 使用 while matrix: 循环来遍历矩阵,直到矩阵为空。
  3. 在每一次循环中,先将矩阵的第一行(即第一层)添加到结果列表 res 中,然后将该行从矩阵中移除。
  4. 接着,将剩下的矩阵逆时针旋转 90 度(将其转置后再沿着竖直方向反转),以便下次循环时能够继续削头。
  5. 重复上述步骤直到矩阵为空。

这种方法的思路简洁,通过不断调整矩阵的结构来实现螺旋顺序遍历,避免了显式的控制遍历方向和边界条件的处理。

在实际代码中,matrix.pop(0) 用于削头,list(zip(*matrix))[::-1] 用于逆时针旋转矩阵。

如果是C++实现类似削头的效果:

#include <vector>

class Solution {
public:
    vector<int> spiralOrder(vector<vector<int>>& matrix) {
        vector<int> res;
        while (!matrix.empty()) {
            // 削头(第一层)
            for (auto it = matrix[0].begin(); it != matrix[0].end(); ++it) {
                res.push_back(*it);
            }
            matrix.erase(matrix.begin()); // 删除第一行

            // 将剩下的逆时针转九十度,等待下次被削
            if (!matrix.empty()) {
                vector<vector<int>> new_matrix;
                for (int j = matrix[0].size() - 1; j >= 0; --j) {
                    vector<int> column;
                    for (int i = 0; i < matrix.size(); ++i) {
                        column.push_back(matrix[i][j]);
                    }
                    new_matrix.push_back(column);
                }
                matrix = new_matrix;
            }
        }
        return res;
    }
};

三、螺旋遍历矩阵 简单

LCR 146. 螺旋遍历二维数组

上面这个题是企业题,同类型的题目可以多练习练习~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/496974.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Redis入门到实战-第十七弹

Redis实战热身t-digest篇 完整命令参考官网 官网地址 声明: 由于操作系统, 版本更新等原因, 文章所列内容不一定100%复现, 还要以官方信息为准 https://redis.io/Redis概述 Redis是一个开源的&#xff08;采用BSD许可证&#xff09;&#xff0c;用作数据库、缓存、消息代理…

2024最新华为OD机试试题库全 -【二叉树计算】- C卷

1. 🌈题目详情 1.1 ⚠️题目 给出一个二叉树如下图所示: 请由该二叉树生成一个新的二叉树,它满足其树中的每个节点将包含原始树中的左子树和右子树的和。 左子树表示该节点左侧叶子节点为根节点的一颗新树;右子树表示该节点右侧叶子节点为根节点的一颗新树。 1.2 �…

自养号测评,补单的五大关键要素

现在很多大卖都是自己管理几百个账号&#xff0c;交给服务商不是特别靠谱 第一 你不知道服务商账号质量怎么样 第二 账号一天下了多少你也不清楚&#xff0c;如果下了很多单万一封号被关联了怎么办 第三 你也不知道服务商用什么卡给你下单&#xff0c;用一些低汇率和黑卡下单…

运用开关量信号远程传输装置实现工厂智能化技改需要分几步走

DTD509H系列开关量信号无线传输器由一个无线信号发射终端和一个无线信号接收终端组成&#xff0c;也可以根据用户现场实现一点对多点或者多点对一点的无线控制。DTD509H系列开关量信号无线传输器可与PLC的IO端口、继电器、二次仪表、传感器等工业设备配套使用&#xff0c;运用无…

【前端Vue】社交信息头条项目完整笔记第2篇:二、登录注册,准备【附代码文档】

社交媒体-信息头条项目完整开发笔记完整教程&#xff08;附代码资料&#xff09;主要内容讲述&#xff1a;一、项目初始化使用 Vue CLI 创建项目,加入 Git 版本管理,调整初始目录结构,导入图标素材,引入 Vant 组件库,移动端 REM 适配,关于 , 配置文件,封装请求模块。十、用户关…

快速上手Spring Cloud 十四:璀璨物联网之路

快速上手Spring Cloud 一&#xff1a;Spring Cloud 简介 快速上手Spring Cloud 二&#xff1a;核心组件解析 快速上手Spring Cloud 三&#xff1a;API网关深入探索与实战应用 快速上手Spring Cloud 四&#xff1a;微服务治理与安全 快速上手Spring Cloud 五&#xff1a;Spring …

Java项目:78 springboot学生宿舍管理系统的设计与开发

作者主页&#xff1a;舒克日记 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文中获取源码 项目介绍 系统的角色&#xff1a;管理员、宿管、学生 管理员管理宿管员&#xff0c;管理学生&#xff0c;修改密码&#xff0c;维护个人信息。 宿管员管理公寓…

iOS - Runtime-API

文章目录 iOS - Runtime-API1. Runtime应用1.1 字典转模型1.2 替换方法实现1.3 利用关联对象给分类添加属性1.4 利用消息转发机制&#xff0c;解决方法找不到的异常问题 2. Runtime-API2.1 Runtime API01 – 类2.1.1 动态创建一个类&#xff08;参数&#xff1a;父类&#xff0…

轻松赚钱,精彩生活:上班族副业赚钱新攻略大揭秘!

薪水总是捉襟见肘&#xff0c;每月账单总让人倍感压力。你是否曾在静谧的夜晚&#xff0c;躺在床上&#xff0c;思索如何为家庭多赚一分钱&#xff1f;其实&#xff0c;你并不孤单。在这个充满机遇与挑战的时代&#xff0c;越来越多的人开始寻找副业&#xff0c;以期望让生活更…

Appium设备交互API

设备交互API指的是操作设备系统中的一些固有功能&#xff0c;而非被测程序的功能&#xff0c;例如模拟来电&#xff0c;模拟发送短信&#xff0c;设置网络&#xff0c;切换横竖屏&#xff0c;APP操作&#xff0c;打开通知栏&#xff0c;录屏等。 模拟来电 make_gsm_call(phon…

八数码问题

八数码难题 题目描述 在 3 3 3\times 3 33 的棋盘上&#xff0c;摆有八个棋子&#xff0c;每个棋子上标有 1 1 1 至 8 8 8 的某一数字。棋盘中留有一个空格&#xff0c;空格用 0 0 0 来表示。空格周围的棋子可以移到空格中。要求解的问题是&#xff1a;给出一种初始布局…

Java项目:79 springboot海滨体育馆管理系统的设计与实现

作者主页&#xff1a;源码空间codegym 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文中获取源码 项目介绍 体育馆管理系统主要实现了管理员功能模块和学生功能模块两大部分 管理员功能模块&#xff1a; 管理员登录后可对系统进行全面管理操作&#…

内网穿透_ICMP_icmpsh

目录 一、ICMP协议详解 二、ICMP隧道 (一) 为什么会使用ICMP (二) 实验环境 (三) 操作流程 1. 下载icmpsh 2. 下载并安装依赖 3. 关闭本地icmp响应 4. 攻击机启动服务端开始监听 5. 靶机启动工具客户端 6. 攻击机接受到靶机传来的数据 三、郑重声明 一、ICMP协议详…

LeetCode_1.两数之和

一、题目描述 二、方法 1.方法1&#xff08;暴力枚举法&#xff09; 利用两个for循环&#xff0c;对数组进行逐一的遍历&#xff0c;直到找到两个数的和为目标值时返回这两个数的下标。以下为c实现的完整代码。 # include<iostream> using namespace std; #include<…

C语言分支循环语句详解

分支和循环语句是什么 在我们写程序的时候&#xff0c;总会遇到想一直循环执行某种语句的时候&#xff0c;这时候我们就要使用一种语句叫循环语句&#xff0c;或者带一些判断条件的语句&#xff0c;在C语言中提供了像这些的循环语句和分支语句 if else 语句 这是一种判断语句…

AcWing 800. 数组元素的目标和(哈希)

原题链接 哈希思路: 我们可以在输入 时把每个数存进哈希表里&#xff0c;对于每个输入的 b[i]看看 x−b[i]是否出现与哈希表即可。 图解 #include <iostream> #include <algorithm> #include <unordered_map> using namespace std;const int N 111111;in…

使用 Yoda 和 ClickHouse 进行实时欺诈检测

背景 Instacart 是北美领先的在线杂货公司,拥有数百万活跃的客户和购物者。在其平台上打击欺诈和滥用行为不仅对于维护一个值得信赖和安全的环境至关重要,也对保持Instacart的财务健康至关重要。在这篇文章中,将介绍了一个欺诈平台——Yoda,解释了为什么我们选择ClickHous…

Game Maker 更新|在 The Sandbox 使用烹饪模拟器!

我们很高兴向你介绍 Game Maker 的最新更新&#xff1a;烹饪模拟器模板&#xff01;让自己沉浸在令人兴奋的新游戏类型中&#xff0c;同时学习有趣的新机制。剖析和探索模板&#xff0c;考验你的开发技能&#xff01; 什么是烹饪模拟游戏&#xff1f; 近年来&#xff0c;随着《…

【JavaScript】数组 ② ( JavaScript 数组索引 | JavaScript 遍历数组 | 使用 for 循环遍历数组 )

文章目录 一、JavaScript 数组索引1、数组索引2、数组索引 - 代码示例 二、JavaScript 遍历数组1、使用 for 循环遍历数组2、使用 for 循环遍历数组 - 代码示例 一、JavaScript 数组索引 1、数组索引 在 JavaScript 中 , 数组 的 " 索引 " 又称为 " 下标 "…

考研数学|武忠祥学习包搭配《660》和《880》

一、660、880、三大计算简单分析 660题 这本题册具有高难度、综合度和深度&#xff0c;属于高质量的题材。我建议不要在基础阶段就着手解决其中的660题&#xff0c;因为这可能会影响你的信心。相反&#xff0c;你可以在基础阶段完成一轮学习后&#xff0c;将这些题目留到强化…