顺时针打印矩阵

面试题29. 顺时针打印矩阵

这题完全不会。。。

参考答案1解题思路:
用一个矩阵保存已访问过的元素(visited),用于判断下一个元素是否已经保存到order数组中
用一个矩阵来秒数下一次访问的方向(directions),其中的每个一维数组中的元素表示的行和列下一次往哪里走
(以{0,1}为例:0表示下一个访问的元素行下标+0,1表示列下标+1,也就是访问同一行的右边一个元素;同理,{-1,0}表示访问同一列的上一行的一个元素)
nextRow和nextColumn用于判断在当前的方向下,下一个元素能否被访问到(是否会出现数组越界、下标为负等情况),如果访问不到,则需要改变方向,而改变方向只需要取directions矩阵中的下个一维数组,因为directions在定义的时候就是按照顺时针运动方向定义的。

public int[] spiralOrder(int[][] matrix) {

        if (matrix == null || matrix.length == 0 || matrix[0].length == 0) {
            return new int[0];
        }
        int rows = matrix.length, columns = matrix[0].length;
        boolean[][] visited = new boolean[rows][columns];
        int total = rows * columns;
        int[] order = new int[total];
        int row = 0, column = 0;
        int[][] directions = { { 0, 1 }, { 1, 0 }, { 0, -1 }, { -1, 0 } };
        int directionIndex = 0;
        //开始访问矩阵中的每一个元素 第一个元素为matrix[0][0],方向为{0,1}即向右移动
        for (int i = 0; i < total; i++) {
            //创建一个一维数组来保存返回值
            order[i] = matrix[row][column];
            //记录被访问的元素
            visited[row][column] = true;
            //根据访问元素的移动方向确定下一个元素的下标
            int nextRow = row + directions[directionIndex][0], nextColumn = column + directions[directionIndex][1];
            //对上一步得到的下标进行判断,是否需要改变方向
            if (nextRow < 0 || nextRow >= rows || nextColumn < 0 || nextColumn >= columns
                    || visited[nextRow][nextColumn]) {
                //如果需要改变方向则取directions的下一个元素,因为始终为顺时针
                directionIndex = (directionIndex + 1) % 4;
            }
            //确定下一次循环所访问的元素的下标
            row += directions[directionIndex][0];
            column += directions[directionIndex][1];
        }
        return order;
    }

举个实际的栗子:
矩阵[[1,2,3],[4,5,6],[7,8,9],[10,11,12]]
开始时direction为{0,1},会访问1,2,3这三个元素。其中1,2这两个元素不进入if,访问3时,nextColum=2+1>=column=3进入了if(此处对应了数组越界),改变了方向,此时direction为{1,0},directionIndex为1。这时row=row+directions[directionIndex][0]=0+1=1
column=column+directions[directionIndex][1]=2+0=2对应了6这个元素。这时开始访问最右侧(column=2)的元素,并且在访问这一列元素时,column保持为2,row每次循环加1(directions[directionIndex][0]=1)。
当访问到10这个元素时,direction为{0,-1},此时的情况为nextColumn<0(此处对应了数组下标为负)。再次改变了方向,变为{-1,0}即自下而上的访问。当访问到4这个元素时,再次进入if,此时是因为1这个元素已经被访问过了,满足了visited[nextRow][nextColumn]=true,再次转向,回到开始的时候directionIndex=0,方向为{0,1}。

另一种解法相对好理解一些:
即一圈一圈访问,先访问最外面一圈,结束后往里缩一圈,继续访问


图示:注意观察行列之间重叠元素的归属问题,涉及到代码中for循环的边界问题
class Solution {
    public int[] spiralOrder(int[][] matrix) {
if (matrix == null || matrix.length == 0 || matrix[0].length == 0) {
                return new int[0];
            }
            int rows = matrix.length, columns = matrix[0].length;
            int[] order = new int[rows * columns];
            int index = 0;
            int left = 0, right = columns - 1, top = 0, bottom = rows - 1;
            while (left <= right && top <= bottom) {
                //访问top一行
                for (int column = left; column <= right; column++) {
                    order[index++] = matrix[top][column];
                }
                //访问right列
              //此处top+1是因为top行的最后一个元素,和right行的第一个元素是同一个,把这个元素放到top行中,right列的访问从top行的下一行开始
                for (int row = top + 1; row <= bottom; row++) {
                    order[index++] = matrix[row][right];
                }
                if (left < right && top < bottom) {//如果当前这个圈左边界和右边界,上边界和下边界没有重合的话,把外圈元素取出来
                    //访问bottom行
                    //和上面一样。bottom行的最右元素归属于right列,所以需要减一
                    for (int column = right - 1; column > left; column--) {
                        order[index++] = matrix[bottom][column];
                    }
                    //访问left列
                    //bottom行的最左元素归属于left列
                    for (int row = bottom; row > top; row--) {
                        order[index++] = matrix[row][left];
                    }
                }
                //缩小一圈
                left++;
                right--;
                top++;
                bottom--;
            }
            return order;
     }
}

作者:LeetCode-Solution
链接:https://leetcode-cn.com/problems/shun-shi-zhen-da-yin-ju-zhen-lcof/solution/shun-shi-zhen-da-yin-ju-zhen-by-leetcode-solution/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,080评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,422评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,630评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,554评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,662评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,856评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,014评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,752评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,212评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,541评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,687评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,347评论 4 331
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,973评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,777评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,006评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,406评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,576评论 2 349