javascript之活灵活现的Array
btek0718
8年前
<h2><strong>前言</strong></h2> <p>就如同标题一样,这篇文章将会灵活的运行Array对象的一些方法来实现看上去较复杂的应用。</p> <p>大家都知道Array实例有这四个方法:push、pop、shift、unshift。大家也都知道 push + pop实现栈, shift + push实现队列。在这里不讨论什么先进后出、先进先出。但一面这个题将要用到这几个方法。</p> <h2><strong>题目</strong></h2> <p>螺旋矩阵这个名词,在后台语言中可能很熟悉,他是个二维数组,他有什么特点呢?请看下图:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/b1fc4452e5bae87ba3e1d46bb9884abf.gif"></p> <p>以上是一个从外到内的螺旋矩阵,他的排列规则是从外围开始走,一圈一圈绕道最里面,就像一条盘旋着的蛇。</p> <h2><strong>分析与解答</strong></h2> <p>进入正题,今年9月份腾讯校招在线笔试题就有一个螺旋矩阵的 题,传入给定数字n,打印出n*n的螺旋矩阵,当时本菜鸟并没有做出来,时间过后自己在电脑上想,然后恍然大悟明白的其中的奥妙。</p> <p>虽然博主没有记录当时的代码,但是我是首先定义了个n*n的二维数组,得到需要绕几层,比如上面的是2层,然后就循环几次,在内部用四个for循环,分别是上下左右给定义的二维数组插内容,具体的代码就没法上了,反正方法很笨,而且并不是本文章的重点,下面进入本章主题:</p> <p>前几天我在codewars上做题,遇到了个螺旋矩阵的题,它是要求写一个函数,给定一个矩阵二维数组参数,返回一个数组,该数组的元素顺序就是螺旋矩阵的路径。</p> <p>例如:</p> <pre> <code class="language-javascript">function getLinear (spiral) { //...做一些操作 } var arr = [ [1,2,3], [4,5,6], [7,8,9] ] getLinear(arr) // 返回 [1,2,3,6,9,8,7,4,5] </code></pre> <p>上面例子清除的看到getLinear函数是将传入的‘螺旋矩阵’用一维数组按顺序输出了(我也不知道怎么说了,反正就是把这个二维数组像蛇一样盘旋着形成一维数组)</p> <p>看到这个题的第一时间,我就想起了腾讯校招的那个题, 然后博主就用类似的四个for循环写完了,然后提交。这个网站有个功能是,你做完题目可以看别人做的代码,博主小心翼翼的点开答案列表,哇,第一条就深深吸引了我。虽然不记得别人写的源码,但大致是这样的:</p> <pre> <code class="language-javascript">function getLinear(spiral) { var item; var linear = [] while (item = spiral.shift()) { // 上 linear = linear.concat(item) // 右 for (var i = 0; i < spiral.length; i++) { linear.push( spiral[i].pop() ) } // 下 linear = linear.concat( spiral.pop().reverse()) // 左 for (var i = spiral.length - 1; i >= 0; i --) { linear.push(spiral[i].shift()) } } return linear } </code></pre> <p>对于菜鸟级别的我来说,刚开始还有点懵,因为跟我的思维不一样,看了一会才发现其中的奥妙。相比我写的真是好多了,这个代码不需要考虑传入的是否为n*n数组,他可以解析任意数组比如2*3数组等。</p> <p>而且代码绝对简洁,对于有一定基础的来说也很容易懂。</p> <p>如果你有些困惑,就往下看,我的图文解释</p> <pre> <code class="language-javascript">// 上 linear = linear.concat(item) </code></pre> <p>item为二维数组的第一个元素,就是第一个数组,将它移除数组并返回,如下:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/84adeaa3d2ab9e2578cdd686b4643135.png"></p> <p>此行代码后,原数组变成如下:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/c0560cd2ea271ad69e2e724cc1637ba4.png"></p> <p>接下来,我们需要将 5 6 7 加入要返回的数组中,也就是二维数组的每个数组元素的最后一个元素,我们可以用pop获取到:</p> <pre> <code class="language-javascript">// 右 for (var i = 0; i < spiral.length; i++) { linear.push( spiral[i].pop() ) } </code></pre> <p>这时原来的二维数组变成了如下:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/91fe49f9179fc0cac64b8ef6052137a5.png"></p> <p>接下来我们要拿到最后一行 10 9 8 并倒置,将二维数组pop出最后一个数组然后将他reverse就可以了</p> <pre> <code class="language-javascript">// 下 linear = linear.concat(spiral.pop().reverse()) </code></pre> <p>此时原来二维数组是这样的:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/b501867ed19536bee588013d6b463e06.png"></p> <p>获取左边的就跟右边类似,只需要将pop变成shift:</p> <pre> <code class="language-javascript">// 左 for (var i = spiral.length - 1; i >= 0; i --) { linear.push(spiral[i].shift()) } </code></pre> <p>原来二维数组变为:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/f47232f9bfb195f242c5d152ceae6d3f.png"></p> <p>此时,一圈就完了,然后while判断是否进入下一圈。</p> <p>本章就到这里了,一个看上去不简单的题目,在Array的灵活下变得如此简单,博主忍不住要分享给js新人们。</p> <p>如果你有更好更简便的方法,可以分享出来我们一起围观~~</p> <p> </p> <p>来自:http://www.cnblogs.com/xujiazheng/p/6113215.html</p> <p> </p>