leetcode-54. Spiral Matrix - Medium

descrition

GGiven a matrix of m x n elements (m rows, n columns), return all elements of the matrix in spiral order.

For example,
Given the following matrix: [
[ 1, 2, 3 ],
[ 4, 5, 6 ],
[ 7, 8, 9 ]
]
You should return [1,2,3,6,9,8,7,4,5].

解析

基本思路是按照规则一个个的取数,两份方法的区别在于处理的技巧不同。

方法 1:

小技巧(si,sj),(ei, ej),对角线可以唯一确定一个矩阵。循环获取结果,每次循环获取当前最外层的一圈数字。

  1. 从左到右,matrix[si][sj,...,ej]
  2. 从上到下,matrix[si,...,ej][ej]
  3. 从右到左,matrix[ei][ej-1,...,sj]
  4. 从下到上,matrix[ei-1,...,si-1][sj]

实现如代码所示,注意边界条件。

方法 2:

我们只需要模拟时钟的顺时针行走,每走一步获取一个元素,假设矩阵 matrix 有 m 行 n 列,那么只需要循环 m * n 次即可获取所有元素。算法描述如下:

定义两个辅助数组变量如下,dr 和 dc 分别表示行和列的方向变化。

dr={0, 1, 0, -1}
dc={1, 0, -1, 0}

两个数组共同组合有以下 4 种情况:

  1. dr[0], dc[0]:行的增量为 0,列的增量为 1,对应方法 1 的情况 1
  2. dr[1], dc[1]:行的增量为 1,列的增量为 0,对应方法 1 的情况 2
  3. dr[2], dc[2]:行的增量为 0,列的增量为 -1, 对应方法 1 的情况 3
  4. dr[3], dc[3]:行的增量为 -1,列的增量为 0,对应方法 1 的情况 4

我们总共遍历 m * n 次,每次获取一个元素,而没个元素的状态都将是以上 4 种状态中的一种。令 r,c 分别表示当前的行和列的下标,cr, cc 分别表示 candidate,即下一个候选位置的下标,di 表示当前的方向(对应以上 4 种情况,即 dr,dc 的下标),visited[][] 对应 matrix,表示元素是否被访问过。

对于任意一次遍历,若 r,c 合法,即没有超出矩阵边界又没有被访问过,那么我们可以获取当前元素。否则,我们需要调换方向,即更新 di 的值,具体实现参考代码。

实际上方法 2 不是最优解,相比于方法 1, 方法 2 需要一个辅助的数组,但其思想非常值得借鉴,尤其是状态控制的 dr[] 和 dc[] ,di

leetcode-solution

code

#include <iostream>
#include <vector>
#include <algorithm> using namespace std; class Solution{
public:
vector<int> spiralOrder(vector<vector<int> >& matrix){ //return spiralOrderCycle(matrix);
return spiralOrderSimilation(matrix); } // time-O(m*n), space-O(1) exclude the space used to return info.
vector<int> spiralOrderCycle(vector<vector<int> >& matrix){
vector<int> ans;
if(matrix.empty())
return ans; int rows = matrix.size(), cols = matrix[0].size();
int si = 0, sj = 0;
int ei = rows - 1, ej = cols - 1; while(si <= ei && sj<=ej){
spiralOrderOneCycle(matrix, si, sj, ei, ej, ans);
si++;
sj++;
ei--;
ej--;
} return ans;
} void spiralOrderOneCycle(vector<vector<int> >& matrix,
int si, int sj, int ei, int ej,
vector<int>& ans){
// top row
for(int j=sj; j<=ej; j++)
ans.push_back(matrix[si][j]); // right column
for(int i=si+1; i<=ei; i++)
ans.push_back(matrix[i][ej]); // down row if exist
if(si < ei){
for(int j=ej-1; j>=sj; j--)
ans.push_back(matrix[ei][j]);
} // left column if exist
if(sj < ej){
for(int i=ei-1; i>si; i--)
ans.push_back(matrix[i][sj]);
}
} // time-O(m*n), space=O(m*n)
vector<int> spiralOrderSimilation(vector<vector<int> > &matrix){
vector<int> ans;
if(matrix.empty())
return ans;
int rows = matrix.size(), cols = matrix[0].size();
vector<vector<bool> > visited(rows, vector<bool>(cols, false));
int dr[] = {0, 1, 0, -1};
int dc[] = {1, 0, -1, 0};
// start from the left-top corner (r,c) = (0,0) and from left to right
// di = 0, dr[0] and dc[0], correspond left to right.
int di = 0, r = 0, c = 0; for(int i=0; i<rows*cols; i++){
ans.push_back(matrix[r][c]);
visited[r][c] = true;
int cr = r + dr[di];
int cc = c + dc[di];
if(0 <= cr && cr < rows && 0 <= cc && cc < cols && !visited[cr][cc]){
r = cr;
c = cc;
}else{
// change direction
di = (di + 1) % 4;
r = r + dr[di];
c = c + dc[di];
}
} return ans;
}
}; int main()
{
return 0;
}

[array] leetcode - 54. Spiral Matrix - Medium的更多相关文章

  1. Leetcode 54. Spiral Matrix & 59. Spiral Matrix II

    54. Spiral Matrix [Medium] Description Given a matrix of m x n elements (m rows, n columns), return ...

  2. LeetCode - 54. Spiral Matrix

    54. Spiral Matrix Problem's Link ------------------------------------------------------------------- ...

  3. leetcode 54. Spiral Matrix 、59. Spiral Matrix II

    54题是把二维数组安卓螺旋的顺序进行打印,59题是把1到n平方的数字按照螺旋的顺序进行放置 54. Spiral Matrix start表示的是每次一圈的开始,每次开始其实就是从(0,0).(1,1 ...

  4. Leetcode 54:Spiral Matrix 螺旋矩阵

    54:Spiral Matrix 螺旋矩阵 Given a matrix of m x n elements (m rows, n columns), return all elements of t ...

  5. leetCode 54.Spiral Matrix(螺旋矩阵) 解题思路和方法

    Spiral Matrix Given a matrix of m x n elements (m rows, n columns), return all elements of the matri ...

  6. LeetCode: 54. Spiral Matrix(Medium)

    1. 原题链接 https://leetcode.com/problems/spiral-matrix/description/ 2. 题目要求 给定一个二维整型数组,返回其螺旋顺序列表,例如: 最后 ...

  7. LeetCode 54. Spiral Matrix(螺旋矩阵)

    Given a matrix of m x n elements (m rows, n columns), return all elements of the matrix in spiral or ...

  8. [leetcode]54. Spiral Matrix螺旋矩阵

    Given a matrix of m x n elements (m rows, n columns), return all elements of the matrix in spiral or ...

  9. [leetcode]54. Spiral Matrix二维数组螺旋取数

    import java.util.ArrayList; import java.util.List; /** * Given a matrix of m x n elements (m rows, n ...

随机推荐

  1. Redis 持久化之RDB和AOP

    Redis 持久化之RDB和AOP Redis 有两种持久化方案,RDB (Redis DataBase)和 AOP (Append Only File).如果你先快速了解和使用RDB和AOP,可以直 ...

  2. PL/SQL 游标 (实验七)

    PL/SQL 游标 emp.dept 目标表结构及数据 要求 基于部门表建立游标dept_cursor1,使用记录变量接收游标数据,输出部门表信息: 显示格式: 部 门 号: XXX 部门名称: XX ...

  3. LSF-SCNN:一种基于 CNN 的短文本表达模型及相似度计算的全新优化模型

    欢迎大家前往腾讯云社区,获取更多腾讯海量技术实践干货哦~ 本篇文章是我在读期间,对自然语言处理中的文本相似度问题研究取得的一点小成果.如果你对自然语言处理 (natural language proc ...

  4. php的定界符<<<eof的问题

    在php的编程过程中难免会遇到输出大段的html和javascript脚本的情况,可都放在具体的地方的时候,路由不好处理,而且比较浪费时间 如果按照传统的输出方法,按照字符串输出的话,需要大量的转义字 ...

  5. springboot集成swagger

    对于搬砖的同学来说,写接口容易,写接口文档很烦,接口变动,维护接口文档就更更更烦,所以经常能发现文档与程序不匹配. 等过一段时间就连开发者也蒙圈了 Swagger2快速方便的解决了以上问题.一个能与S ...

  6. JavaScript:彻底理解同步、异步和事件循环(Event Loop)

    一. 单线程 我们常说"JavaScript是单线程的". 所谓单线程,是指在JS引擎中负责解释和执行JavaScript代码的线程只有一个.不妨叫它主线程. 但是实际上还存在其他 ...

  7. Mac关机时处于黑屏状态

    PS:不知道大家有没有遇到过mac电脑关机就黑屏,只有一个箭头,还可以滑动箭头,但就是黑屏状态,等个好长时间还是关不了机,因此我查了好多资料,原因是在关机时,mac要先关掉其他软件或者保存进程以备下次 ...

  8. ACM HDU 1081 To The Max

     To The Max Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) To ...

  9. HDU 3549 Flow Problem 网络流(最大流) FF EK

    Flow Problem Time Limit: 5000/5000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others) Tot ...

  10. PHP基础 windows环境下安装Apache Mysql PHP

    本篇文章主要是讲一下我自己安装wamp环境的一些步骤和见解,前方多图预警,慎入!!!!! PHP运行环境  : Linux下的三种安装方式:源码包安装.rpm包安装.集成环境安装(lnmp) wind ...