有一个n行m列的矩阵(1≤n≤1000,1≤m≤1000),在这个矩阵上进行q (1≤q≤100,000) 个操作: 1 x y: 交换矩阵M的第x行和第y行(1≤x,y≤n); 2 x y: 交换矩阵M的第x列和第y列(1≤x,y≤m); 3 x y: 对矩阵M的第x行的每一个数加上y(1≤x≤n,1≤y≤10,000); 4 x y: 对矩阵M的第x列的每一个数加上y(1≤x≤m,1≤y≤10,000);
输入包含多组数据. 第一行有一个整数T(1≤T≤15), 表示测试数据的组数. 对于每组数据: 第一行输入3个整数n, m, q. 接下来的n行,每行包括m个整数,表示矩阵M。(1≤Mi,j≤10,000),(1≤i≤n,1≤j≤m). 最后q行,每行输入三个整数a(1≤a≤4), x, y。
对于每组数据,输出经过所有q个操作以后的矩阵M。
2 3 4 2 1 2 3 4 2 3 4 5 3 4 5 6 1 1 2 3 1 10 2 2 2 1 10 10 1 1 1 2 2 1 2
12 13 14 15 1 2 3 4 3 4 5 6 1 10 10 1
建议使用scanf / printf 代替 cin / cout
思路:
这题一开始是想到直接交换的,但是因为证明不到正确性而且但是不知道哪里找了一个反例推翻了当前的想法,所以就认为想法错了,就另外找个办法吧!不过这题我在交换时是用异或做交换,结果Wa了,换了swap就过了,诡异。
Matrix
对于交换行、交换列的操作,分别记录当前状态下每一行、每一列是原始数组的哪一行、哪一列即可。
对每一行、每一列加一个数的操作,也可以两个数组分别记录。注意当交换行、列的同时,也要交换增量数组。
输出时通过索引找到原矩阵中的值,再加上行、列的增量。
复杂度O(q+mn)
AC代码:
#include<iostream> #include<algorithm> #include<cstring> #include<string> #include<vector> using namespace std; typedef long long ll; #define T 1000 + 50 #define inf 0x3f3f3f3f struct jz { int M[T][T]; int row[T][2],col[T][2]; }; jz p; int main() { #ifdef zsc freopen("input.txt","r",stdin); #endif int n,m,q,i,j,k,t; scanf("%d",&t); while(t--) { scanf("%d%d%d",&n,&m,&q); for(i=1;i<=n;++i){ for(j=1;j<=m;++j){ p.row[i][0] = i;p.row[i][1] = 0; p.col[j][0] = j;p.col[j][1] = 0; scanf("%d",&p.M[i][j]); } } int c,x,y; while(q--) { scanf("%d%d%d",&c,&x,&y); if(c==1){ swap(p.row[x][0],p.row[y][0]); } else if(c==2){ swap(p.col[x][0],p.col[y][0]); } else if(c==3){ p.row[p.row[x][0]][1] += y; } else { p.col[p.col[x][0]][1] += y; } } for(i=1;i<=n;++i){ for(j=1;j<=m;++j){ printf("%d",p.M[p.row[i][0]][p.col[j][0]] +p.row[p.row[i][0]][1]+p.col[p.col[j][0]][1]); if(j!=m)printf(" "); } printf("\n"); } } return 0; }