题目描述

输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字。
例如,如果输入如下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.
 
 
思想:
一圈一圈的打印,这道题难点在于何时中止打印。
 
图为在矩阵中某一圈,length为矩阵长度,width为矩阵宽度,坐标值为数组下标。

对于我们这道题来说,只要有内圈可以进入下一圈就是以第一圈(0,0)    第二圈 (1,1) ……的规律做每一圈的起点

现在考虑以下标(i,i)为起点的一圈,其图像为length为矩阵长度,width为矩阵宽度

打印过程是四个for循环以此为上-》右-》下-》左顺时针遍历

按照图我们开始遍历第i圈

//注意四个角的元素不能重复
//上 行i 列i length-i-1
for (int j=i; j <= length-i-1; j++) {
result.push_back(matrix[i][j]);
}
//右 列 length-i-1 行i+1
for (int j= i+1; j <= width - i - 1; j++) {
result.push_back(matrix[j][length-i-1]);
}
//下 行width-i-1 列length-2-1 到i
for (int j = length-i-2; j >= i; j--) {
result.push_back(matrix[width-i-1][j]);
}
//左 列i 行 width-i-2 到 i+1
for (int j = width-i-2; j > i; j--) {
result.push_back(matrix[j][i]);
}

  

  

现在一圈完了,下一圈会不会有什么突发状况?

思考:假设下一圈就结束循环了,那么可能遇到什么情况,下一圈不再是一个完整矩形(指的是长宽都大于1的矩阵)而是有四种可能。我们之所以分析是要看这几种情况对我们四个for循环是否都适用

1.没有元素

2.只有一个元素

3.只有一行元素

4.只有一列元素

好的,现在我们知道了所有可能的形状。那进入到这几种情况我们就要考虑了。

1.本圈矩形的长宽只要任意一个为0就结束循环,没有元素了 条件 宽0长0   length-2i<=0或width-2i<=0  

此时直接跳出循环即可,四个for循环都不满足

2.只有一个元素和只有一行元素  就是我们只需要进行 顺势针打印的上面部分,此时条件是 宽1长若干  (width-2i==1且length-2i大于1)化简此时 width=2i-i

//打印一行没有问题,按道理来说是不是之后的打印都不应该做

观察下

右面打印的判断条件   int j= i+1; j <= width - i - 1; j++   因为宽度为1  初始    width为2i+1   代入for循环判断不满足   j=i+1<  i条件

下面打印的判断条件   int j = length-i-2; j >= i; j--   假设length无穷大,那可进入for循环从左到右打印

左面打印的判断条件类似右面,因宽度为1本身就可限制

总结:因长度大于1,宽度等于1出现重复打印情况如下:

所以我们在   顺时针上面对应的for结束,右侧下面左侧的for可以避免。但是我们下侧for循环就需要有判断条件,为   if(宽度为1则下侧for循环不执行)

3.只有一行元素  就是我们可以顺时针打印 上面 和 左面的部分,此时条件是 宽若干长1  (width-2i大于1且length-2i==1)

类似一行的情况,要给左侧for循环加判断条件  if(长度为1可以不打左侧,因为右侧打完了)

ok,那基于上面的分析我们知道了只要下一圈有元素,我们就可以i++,四个for遍历下一圈,加上我们的if条件。即使出现特殊情况也不必担心。

那下一圈有元素的条件就是非没有元素的条件(length-2*i>0)&&(width-2*i>0)

综上所述,总结一下 对于单行、单列要考虑左右、上下打印条件的相互制约。

给出代码和运行截图:

 class Solution {
public:
static vector<int> printMatrix(vector<vector<int> > matrix) {
if (matrix.size() == ) return vector<int>();
int width = matrix.size(), length = matrix[].size();//r,c为矩阵长宽
int i = ;
vector<int> result;
while ((length - * i > ) && (width - * i > ))
{
//上侧
for (int j = i; j <= length - i - ; j++) {
result.push_back(matrix[i][j]);
} //右侧
for (int j = i + ; j <= width - i - ; j++) {
result.push_back(matrix[j][length - i - ]);
}
//下侧
if (width - * i != ) {//宽度为1不要有右到左打印一遍了,上侧已经打了
for (int j = length - i - ; j >= i; j--) {
result.push_back(matrix[width - i - ][j]);
}
}
//左侧
if (length - * i != ) {//长度为1不要再由下到上打印一遍了,右侧从上到下打了
for (int j = width - i - ; j > i; j--) {
result.push_back(matrix[j][i]);
}
}
i++; }
return result;
} };

运行截图:

附上一个小知识点:vectot<int> 的push_back函数会像数组尾部填数据。预先申请好大小后vector<int> a(5); 用 push_back时会扩容的去填,前面的5个数都是0。

我误认为要先申请空间才能向里添加数据了。

最后,感谢阅读文章,如有疏漏,欢迎指正。

现在转了一圈,要结束了,结束时会出现四种可能,一行、一列、一个、无元素

对于一行的情况,其条件是只

剑指Offer(十九)——顺时针打印矩阵的更多相关文章

  1. 剑指offer 20:顺时针打印矩阵

    题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数 ...

  2. 剑指offer十九之顺时针打印矩阵

    一.题目 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数字1,2, ...

  3. 剑指offer系列17---顺时针打印矩阵(不是很懂)

    package com.exe4.offer; import java.util.ArrayList; /** * 17[题目]输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果 ...

  4. 剑指offer-面试题20.顺时针打印矩阵

    题目:输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字.例如: 输入一个矩阵如下: 则依次打印出数字:1,2,3,4,8,12,16,15,14,13,9,5,6,7,11,10 这道题的 ...

  5. 剑指offer系列18---顺时针打印矩阵

    [题目]定义栈的数据结构,请在该类型中实现一个能够得到栈最小元素的min函数.要求时间复杂度是O(1).push pop min [思路]设计一个辅助栈,当新加入的原数据栈中的数小于辅助栈顶的数时就加 ...

  6. 【剑指Offer】把二叉树打印成多行 解题报告(Python)

    [剑指Offer]把二叉树打印成多行 解题报告(Python) 标签(空格分隔): 剑指Offer 题目地址:https://www.nowcoder.com/ta/coding-interviews ...

  7. Go语言实现:【剑指offer】把二叉树打印成多行

    该题目来源于牛客网<剑指offer>专题. 从上到下按层打印二叉树,同一层结点从左至右输出.每一层输出一行. 需要分层,二维数组. Go语言实现: /** * Definition for ...

  8. 剑指offer-面试题29-顺时针打印矩阵-矩阵

    /* 题目: 输入一个矩阵,按照从外到内顺时针的顺序依次打印每一个数字. */ /* 思路: 1.将打印矩阵看作是打印一个个从外向内的环. 2.每一个环都有一个起始节点,起始节点的坐标*2小于行数和列 ...

  9. 剑指offer系列33-----把二叉树打印成多行

    [题目]从上到下按层打印二叉树,同一层结点从左至右输出.每一层输出一行. 方法一:直接打印 package com.exe7.offer; import java.util.LinkedList; i ...

  10. 算法学习之剑指offer(九)

    一 题目描述 求1+2+3+...+n,要求不能使用乘除法.for.while.if.else.switch.case等关键字及条件判断语句(A?B:C). public class Solution ...

随机推荐

  1. sudo apt-get install 、 pip install和conda install的对比

    sudo apt-get install: apt-get可以用来安装软件.更新源,也可以用来更新自Ubuntu的典型依赖包. (sudo apt-get remove --purge 软件名称 su ...

  2. IDEA微服务项目的application.yml没有绿色叶子的解决办法

    1.今天在写微服务项目的时候成功入坑,那么问题是啥呢?接下来和我一起走入bug的世界吧,让我们看看究竟是怎么回事. *问题描述 1.application.yml是灰色的小格子 2.实在难看 *需要解 ...

  3. 洛谷 P3868 [TJOI2009]猜数字

    题意简述 给定\(a[1],a[2],\cdots,a[n]\) 和 \(b[1],b[2],\cdots,b[n]\),其中\(b\)中元素两两互素. 求最小的非负整数\(n\),满足对于任意的\( ...

  4. [VB.NET Tips]ParamArray参数数组

    ParamArray参数数组,可以理解为传递给方法的多余的参数全都存放在这个数组中. ParamArray只能是ByVal按值传递,不能是可选参数,而且只能做为方法定义的最后一个参数. 非常类似于Py ...

  5. 睡梦中被拉起来执行Spring事务

    梦中惊醒 在Tomcat的线程池里,有这样一个线程,自打出生后,从来不去干活儿,有好多次走出线程池“这座大山”去看世界的机会,都被他拱手让给了弟兄们. 弟兄们给他取了个名字叫二师兄.没错,好吃懒做,饱 ...

  6. ASP.NET Core 3.0 使用gRPC

    一.简介 gRPC 是一个由Google开源的,跨语言的,高性能的远程过程调用(RPC)框架. gRPC使客户端和服务端应用程序可以透明地进行通信,并简化了连接系统的构建.它使用HTTP/2作为通信协 ...

  7. 记一次处理mysql数据库无故锁表的经历

    某日,生产环境上的用户表突然无故锁表,原以为只是偶发的bug.所以第一时间想到的解决方案简单粗暴:重启数据库(service mysqld restart).问题得以解决. 10min后,该表再次锁表 ...

  8. Centos利用脚本自动安装jdk

        在工作中还有自己的学习中,无论是使用tar包安装jdk,还是使用rpm安装,如果单台机器还能够接受,但是如果多台机器,就很困扰.所以,在自己配置环境的时候,根据网上各位前辈,沉淀了这样子一个脚 ...

  9. 谁是狸猫谁是太子?--戏说java构造器

    故事背景 <狸猫换太子>在我国民间文学中很出名,故事剧情大致如下:北宋第三位皇帝宋真宗赵恒年长无子,他的两个妃子刘妃与李妃同时怀了身孕.真宗召见二人,各赐信物,并声明哪个生了儿子就立谁为皇 ...

  10. 环境搭建-Hadoop集群搭建

    环境搭建-Hadoop集群搭建 写在前面,前面我们快速搭建好了centos的集群环境,接下来,我们就来开始hadoop的集群的搭建工作 实验环境 Hadoop版本:CDH 5.7.0 这里,我想说一下 ...