剑指offer - 顺时针打印矩阵 - JavaScript
题目描述
输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下 4 X 4 矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数字 1,2,3,4,8,12,16,15,14,13,9,5,6,7,11,10.
解法 1: 模拟路径
根据直觉,当遍历的过程中,遇到超出边界 / 元素已经被访问过的情况时,应该按照顺时针转变方向。
假设给定的矩阵的形状是 m*n,那么一共要遍历 m*n 次。要准备一个长度为 m*n 的哈希表,来保存元素是否被遍历过。要准备一个记录方向的数组,里面方向的排列顺序是顺时针。
时间复杂度为 O(M*N),空间复杂度为 O(M*N)。
代码实现如下:
// ac地址:https://www.nowcoder.com/practice/9b4c81a02cd34f76be2659fa0d54342a
// 原文地址:https://xxoo521.com/2020-01-30-shun-shi-zhen-matrix/
/**
- @param {number} i
- @param {number} j
*/
function hash(i, j) {
return <span class="hljs-subst" style="color: #333; font-weight: normal; line-height: 26px;">${i}</span>-<span class="hljs-subst" style="color: #333; font-weight: normal; line-height: 26px;">${j}</span>
;
}
/**
- @param {number[][]} matrix
- @return {number[]}
*/
function printMatrix(matrix) {
const m = matrix.length;
if (!m) {
return [];
}
const n = matrix[0].length;
if (!n) {
return [];
}
const results = []; // 遍历结果
const visited = {}; // 记录元素是否被访问过
const directions = [
[0, 1],
[1, 0],
[0, -1],
[-1, 0]
]; // 顺时针方向数组
for (let step = 0, row = 0, col = 0, dIdx = 0; step < m * n; ++step) {
results.push(matrix[row][col]);
visited[hash(row, col)] = true;
// 最巧妙的地方:借助方向数组来进行row、col的更新
const newR = row + directions[dIdx][0];
const newC = col + directions[dIdx][1];
<span class="hljs-keyword" style="color: #333; font-weight: bold; line-height: 26px;">if</span> (
!visited[hash(newR, newC)] &&
newR >= <span class="hljs-number" style="color: #008080; line-height: 26px;">0</span> &&
newR < m &&
newC >= <span class="hljs-number" style="color: #008080; line-height: 26px;">0</span> &&
newC < n
) {
row = newR;
col = newC;
} <span class="hljs-keyword" style="color: #333; font-weight: bold; line-height: 26px;">else</span> {
<span class="hljs-comment" style="color: #998; font-style: italic; line-height: 26px;">// 转变方向</span>
dIdx = (dIdx + <span class="hljs-number" style="color: #008080; line-height: 26px;">1</span>) % <span class="hljs-number" style="color: #008080; line-height: 26px;">4</span>;
row += directions[dIdx][<span class="hljs-number" style="color: #008080; line-height: 26px;">0</span>];
col += directions[dIdx][<span class="hljs-number" style="color: #008080; line-height: 26px;">1</span>];
}
}
return results;
}
解法 2: 按层遍历
这种方法的思路是从外到内,一层层打印。难点在于怎么找到标记点,以及防止重复遍历。
怎么找到标记点?对于每一层来说,设左上角的元素坐标为 (i, j),那么右上角的元素坐标为 (i, n - j - 1),右下角的元素坐标是 (m - i - 1 ,n - j - 1),左下角的元素坐标是 (m - i - 1, j)。找到标记点后,就是对行/列进行+/-的过程。
怎么防止重复遍历?找到四个坐标点后,每一层的遍历可以拆分成 4 个部分。我想到的是下图所示的两种拆分方法:

第一种拆分方法会有例外,比较难处理,无法避免重复遍历。因此使用第二种。
时间复杂度为 O(M*N),空间复杂度为 O(M*N)。代码实现如下:
// ac地址:https://www.nowcoder.com/practice/9b4c81a02cd34f76be2659fa0d54342a
// 原文地址:https://xxoo521.com/2020-01-30-shun-shi-zhen-matrix/
/**
- @param {number[][]} matrix
- @return {number[]}
*/
function printMatrix(matrix) {
const m = matrix.length;
if (!m) {
return [];
}
const n = matrix[0].length;
if (!n) {
return [];
}
const results = [];
let i = 0,
j = 0;
// 这里的终止条件是: i <= (m - 1) / 2 与 j <= (n - j) / 2
// 即最里面的那层左上角元素的坐标
while (i <= m - i - 1 && j <= n - j - 1) {
for (let col = j; col <= n - j - 1; ++col) {
results.push(matrix[i][col]);
}
<span class="hljs-keyword" style="color: #333; font-weight: bold; line-height: 26px;">for</span> (<span class="hljs-keyword" style="color: #333; font-weight: bold; line-height: 26px;">let</span> row = i + <span class="hljs-number" style="color: #008080; line-height: 26px;">1</span>; row <= m - i - <span class="hljs-number" style="color: #008080; line-height: 26px;">1</span>; ++row) {
results.push(matrix[row][n - j - <span class="hljs-number" style="color: #008080; line-height: 26px;">1</span>]);
}
<span class="hljs-keyword" style="color: #333; font-weight: bold; line-height: 26px;">if</span> (i < m - i - <span class="hljs-number" style="color: #008080; line-height: 26px;">1</span> && j < n - j - <span class="hljs-number" style="color: #008080; line-height: 26px;">1</span>) {
<span class="hljs-keyword" style="color: #333; font-weight: bold; line-height: 26px;">for</span> (<span class="hljs-keyword" style="color: #333; font-weight: bold; line-height: 26px;">let</span> col = n - j - <span class="hljs-number" style="color: #008080; line-height: 26px;">2</span>; col > j; --col) {
results.push(matrix[m - i - <span class="hljs-number" style="color: #008080; line-height: 26px;">1</span>][col]);
}
<span class="hljs-keyword" style="color: #333; font-weight: bold; line-height: 26px;">for</span> (<span class="hljs-keyword" style="color: #333; font-weight: bold; line-height: 26px;">let</span> row = m - i - <span class="hljs-number" style="color: #008080; line-height: 26px;">1</span>; row > i; --row) {
results.push(matrix[row][j]);
}
}
i++;
j++;
}
return results;
}
最后
觉得不错,帮忙点个推荐呗,您的支持是对我最大的激励 欢迎我的公众号:「心谭博客」,只专注于前端 + 算法的原创分享
由于个人精力有限,很多系列和历史文章没有即时同步,请前往「前端图谱」&「算法题解」,保证您有所收获。

剑指offer - 顺时针打印矩阵 - JavaScript的更多相关文章
- 剑指Offer 顺时针打印矩阵
题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数字1,2, ...
- 剑指OFFER——顺时针打印矩阵
输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数字1,2,3,4,8 ...
- 剑指Offer顺时针打印矩阵
题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数 ...
- 剑指Offer_Java_顺时针打印矩阵(二维数组)
顺(逆)时针打印矩阵 算法思想: 简单来说,就是不断地收缩矩阵的边界 定义四个变量代表范围,up(初始0).down(初始-行高).left(初始-0).right(初始-列宽), 向右走存入整行的值 ...
- 剑指Offer-19.顺时针打印矩阵(C++/Java)
题目: 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数字 ...
- 剑指offer--26.顺时针打印矩阵
1,2,3,45,6,7,88,10,11,1213,14,15,16 每次输出第一行,然后删除第一行,逆时针旋转剩下的矩阵. ------------------------------------ ...
- python剑指offer 顺时针打印指针
题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数 ...
- 用js刷剑指offer(顺时针打印数组)
题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数 ...
- 剑指Offer18 顺时针打印矩阵
/************************************************************************* > File Name: 18_PrintM ...
随机推荐
- 对Python中列表和数组的赋值,浅拷贝和深拷贝的实例讲解
引用:https://www.jb51.net/article/142775.htm 列表赋值: 1 2 3 4 5 6 7 >>> a = [1, 2, 3] >>&g ...
- Establishing SSL connection without server's identity verification is not recommended. According to MySQL 5.5.45+, 5.6.26+ and 5.7.6+ requirements SSL
由于mysql版本过高创建连接的时候会出现如下报告 解决办法:在mysql连接上加上&useSSL=true 如下:jdbc:mysql:///:3366:test?useUnicode=tr ...
- I Hate it-HDU1754 点修改+区间最大值
题意: 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少.这让很多学生很反感.不管你喜不喜欢, 现在需要你做的是,就是按照老师的要求,写一个程序,模拟老师的询问.当然, ...
- 吴裕雄--天生自然HADOOP操作实验学习笔记:协同过滤算法
实验目的 初步认识推荐系统 学会用mapreduce实现复杂的算法 学会系统过滤算法的基本步骤 实验原理 前面我们说过了qq的好友推荐,其实推荐算法是所有机器学习算法中最重要.最基础.最复杂的算法,一 ...
- 吴裕雄 Bootstrap 前端框架开发——Bootstrap 按钮:用于要弹出信息的按钮
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...
- Python学习笔记之基础篇(五)字典
#数据类型划分:可变数据类型 不可变数据类型 #不可变数据类型 : 元组 bool int str --> 可哈希 #可变数据类型 list ,dict set --->不可哈希 ''' ...
- vagrant 添加带版本号的 box
众所周知,vagrant添加box的时候要从外网下载,那速度...(说多了都是泪),所以只好用下载工具下载到本地之后再添加. 如何搭建 homestead:https://laravelacademy ...
- 第1节 kafka消息队列:1、kafka基本介绍以及与传统消息队列的对比
1. Kafka介绍 l Apache Kafka是一个开源消息系统,由Scala写成.是由Apache软件基金会开发的一个开源消息系统项目. l Kafka最初是由LinkedIn开发,并于20 ...
- 仿照Android标准API写的各种形式的弹出框
strings.xml: <?xml version="1.0" encoding="utf-8"?> <resources> < ...
- SciPy k均值聚类
章节 SciPy 介绍 SciPy 安装 SciPy 基础功能 SciPy 特殊函数 SciPy k均值聚类 SciPy 常量 SciPy fftpack(傅里叶变换) SciPy 积分 SciPy ...