特定顺序遍历二维数组
54. 螺旋矩阵
class Solution {
public List<Integer> spiralOrder(int[][] matrix) {
if(matrix.length==0)
return new ArrayList<>();
int l=0;r=matrix[0].length-1,t=0,b=matrix.length-1;
Integer[] res=new Integer[(r+1)*(b+1)];
while(true){
for(int i=l;i<=r;i++) res[x++]=matrix[t][i];//由左到右
if(++t>b) break; //第一行遍历完了,所以t上边界要下移
for(int i=t;i<=b;i++) res[x++]=matrix[i][r];//由上到下
if(--r<l) break;
for(int i=r;i>=l;i--) res[x++]=matrix[b][i];//由右到左
if(--b<t) break;
for(int i=b;b>=t;i--) res[x++]=matrix[i][l];//由下到上
if(++l>r) break;
}
retuen Arrays.asList(res);
}
}
这道题我参考了作者:Krahets 本题链接链接
我认为这世道很好的题,之后的矩阵只要涉及了顺时针遍历就可以套用这个模板,就算是其他特定顺序也可以参考这个。
59. 螺旋矩阵 II
class Solution {
public int[][] generateMatrix(int n) {
int l=0,r=n-1,t=0,b=n-1,x=1;
int[][] res=new int[n][n];
while(true){
for(int i=l;i<=r;i++) {res[t][i]=x++;}
if(++t>b) break;
for(int i=t;i<=b;i++) {res[i][r]=x++;}
if(--r<l) break;
for(int i=r;i>=l;i--) {res[b][i]=x++;}
if(--b<t) break;
for(int i=b;i>=t;i--) {res[i][l]=x++;}
if(++l>r) break;
}
return res;
}
}
这道题就可以现学现用了,直接顺时针遍历,所有遍历到的元素加1就好了。
498. 对角线遍历
class Solution {
public int[] findDiagonalOrder(int[][] mat) {
int m = mat.length;
int n = mat[0].length;
int[] res = new int[m * n];
int pos = 0;
for (int i = 0; i < m + n - 1; i++) {
if (i % 2 == 1) {
int x = i < n ? 0 : i - n + 1;
int y = i < n ? i : n - 1;
while (x < m && y >= 0) {
res[pos] = mat[x][y];
pos++;
x++;
y--;
}
} else {
int x = i < m ? i : m - 1;
int y = i < m ? 0 : i - m + 1;
while (x >= 0 && y < n) {
res[pos] = mat[x][y];
pos++;
x--;
y++;
}
}
}
return res;
}
}
作者:力扣官方题解
链接:https://leetcode.cn/problems/diagonal-traverse/solutions/1597961/dui-jiao-xian-bian-li-by-leetcode-soluti-plz7/
这道题直接拿官方的了。用了模拟的方法,模拟了每个对角线。
二维数组变换
566. 重塑矩阵
class Solution {
public int[][] matrixReshape(int[][] mat, int r, int c) {
int m=mat.length,n=mat[0].length;
if(m*n!=r*c)
return mat;
int[][] ans=new int[r][c];
for(int i=0;i<m*n;i++){
ans[i/c][i%c]=mat[i/n][i%n];
}
return ans;
}
}
这道题依旧参考官方,因为以我的智商只会暴力求解了。
ans[i/c][i%c]=mat[i/n][i%n];
这一步很精髓。
我的理解是for(int i=0;i<m*n;i++)
这一步意为,把原数组抽象成一维数组,遍历原数组每一个元素。最后映射到二维数组中。
48. 旋转图像
class Solution {
public void rotate(int[][] matrix) {
int n=matrix.length;
int[][] ans=new int[n][n];
for(int i=0,k=0;i<n;i++,k++){
for(int j=n-1,l=0;j>=0;j--,l++)
ans[k][l]=matrix[j][i];
}
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
matrix[i][j] = ans[i][j];
}
}
}
}
这题的话还算简单,但有个关于数组的知识点:
我一开始是使用matrix=ans
给matrix赋值的,但一直赋值上去???传参实际上是传递的是引用的副本,不是引用本身。
这里要明白matrix=ans
是指改变matrix
副本的引用,所以没有改变实际matrix
的值。
而循环可以给matrix
赋值的原因是,matrix
副本仍然是指向实际matrix
的。所以你改变了副本的引用原来的数组不变,而你直接更改副本原来的值就变了。