题目链接

https://codeforces.com/problemset/problem/1980/E

题意

共输入\(T\)组测试用例,每组测试用例第一行输入两个整数\(n, m\),分别代表输入的数据的行数和列数,其中\(1 \leq T \leq 10^4,n * m \leq 2 * 10^5\)。

接下来输入两个\(n\)行\(m\)列的矩阵\(a, b\),对于每个矩阵中的元素\(x_{i,j}\)都是不同的,且满足:\(1 \leq x_{i,j} \leq n * m\)。

你有以下两种可以执行的操作:

  1. 选择两个不同的行进行交换
  2. 选择两个不同的列进行交换

你可以执行上述操作任意次,每次只能选择其中一个操作。问:能否通过上述操作将矩阵\(b\)变为矩阵\(a\),若能,输出“yes”,否则,输出“no”。

题解

本质是考察线性代数的相等矩阵的知识。

矩阵相等的条件:

  1. 两个矩阵为同型矩阵
  2. 矩阵的元素完全相同

对于条件1,由于矩阵\(a, b\)均为\(n\)行\(m\)列的矩阵,故明显成立

对于条件2,只能使用选择两个不同的行/列进行交换,去尝试使之成立

对于初等行/列变换中的行/列交换,有一个非常重要的性质,就是进行行(列)交换后,行(列)内元素的相对位置不会发生变化

基于上述性质,假设此时发生一次行交换,此时行内的相对顺序是不变的,只是每一列元素的相对位置会发生改变,但是发生改变的仅仅是列内元素的相对位置,原本处于同一列的元素依然会处于同一列。

再假设此时发生一次列交换,此时列内的相对顺序是不变的,只是每一行元素的相对位置会发生改变,但是发生改变的仅仅只是行内元素的相对位置,原本处于同一行的元素依然会处于同一行。

基于上述结论,我们只需要判断出矩阵\(a, b\)的每一行、列各自组成的集合,能否在某个顺序下完全相同即可。若能,则输出“YES”,否则,输出“NO”。

在编码实现上,有个问题就是若直接开一个2e5 * 2e5的矩阵,复杂度必然是不可接受的。

观察题目,已知\(n * m \leq 2e5\),那么不妨直接进行矩阵压缩,将二维矩阵压缩为一维,下面讲解如何将二维矩阵压缩为一维数组:

下图是将\(n\)行\(m\)列的二维矩阵映射成长度为\(n * m\)的一维数组:

观察上图,易知原二维矩阵中,处于同一列的元素,在一维数组中是偏移了长度为\(m\)的长度。在二维矩阵中原本同处于第\(i(1 \leq i \leq n)\)行的元素,在一维数组中是从第\((i - 1) * m + 1\)到第\(i * m\)个元素。

因此,若想在一维数组中遍历二维矩阵的每一行,可以用如下代码:

for (int i = 0; i < n; ++ i) {//共n行
for (int j = 0; j < m; ++ j) {//共m列
//i * m + j
}
}

若想在一维数组中遍历二维矩阵的每一列,可以用如下代码:

for (int i = 0; i < m; ++ i) {//共m列
for (int j = 0; j < n; ++ j) {//共n行
//j * m + i
}
}

参考代码

#include<bits/stdc++.h>
#define IOS ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
using namespace std; constexpr int N = 2e5 + 7;
int T, n, m;
int a[N], b[N], row[N], col[N]; void solve() {
cin >> n >> m;
for (int i = 0; i < n * m; ++ i) {
cin >> a[i];
row[a[i]] = i / m;
col[a[i]] = i % m;
}
for (int i = 0; i < n * m; ++ i) cin >> b[i];
if (n == 1 || m == 1) {//行或列等于1时必相等
cout << "YES\n";
return ;
}
//求同行
for (int i = 0; i < n; ++ i) {
int r = row[b[i * m]];
for (int j = 1; j < m; ++ j) {
if (row[b[i * m + j]] != r) {
cout << "NO\n";
return ;
}
}
}
//求同列
for (int i = 0; i < m; ++ i) {
int c = col[b[i]];
for (int j = 1; j < n; ++ j) {
if (col[b[j * m + i]] != c) {
cout << "NO\n";
return ;
}
}
}
cout << "YES\n";
} int main() {
IOS
cin >> T;
while (T --) {
solve();
}
return 0;
}

【矩阵压缩】codeforces 1980 E. Permutation of Rows and Columns的更多相关文章

  1. C++实现矩阵压缩

    C++实现矩阵压缩 转置运算时一种最简单的矩阵运算.对于一个m*n的矩阵M,他的转置矩阵T是一个n*m的矩阵,且T(i,j) = M(j,i). 一个稀疏矩阵的转置矩阵仍然是稀疏矩阵. 矩阵转置 方案 ...

  2. CodeForces 839B - Game of the Rows | Codeforces Round #428 (Div. 2)

    血崩- - /* CodeForces 839B - Game of the Rows [ 贪心,分类讨论] | Codeforces Round #428 (Div. 2) 注意 2 7 2 2 2 ...

  3. [Codeforces 1208D]Restore Permutation (树状数组)

    [Codeforces 1208D]Restore Permutation (树状数组) 题面 有一个长度为n的排列a.对于每个元素i,\(s_i\)表示\(\sum_{j=1,a_j<a_i} ...

  4. 暴力+构造 Codeforces Round #283 (Div. 2) C. Removing Columns

    题目传送门 /* 题意:删除若干行,使得n行字符串成递增排序 暴力+构造:从前往后枚举列,当之前的顺序已经正确时,之后就不用考虑了,这样删列最小 */ /*********************** ...

  5. #467 – 使用UniformGrid 均分行和列(Use a UniformGrid for Evenly Spaced Rows and Columns)

    原文 #467 – 使用UniformGrid 均分行和列(Use a UniformGrid for Evenly Spaced Rows and Columns) UniformGrid 布局面板 ...

  6. CodeForces 483C Diverse Permutation

    Diverse Permutation Time Limit:1000MS     Memory Limit:262144KB     64bit IO Format:%I64d & %I64 ...

  7. [Swust OJ 589]--吃西瓜(三维矩阵压缩)

    题目链接:http://acm.swust.edu.cn/problem/589/ Time limit(ms): 2000 Memory limit(kb): 65535   Description ...

  8. Codeforces 839B Game of the Rows【贪心】

    B. Game of the Rows time limit per test:1 second memory limit per test:256 megabytes input:standard ...

  9. Codeforces 839B Game of the Rows - 贪心

    Daenerys Targaryen has an army consisting of k groups of soldiers, the i-th group contains ai soldie ...

  10. Poj 3318 Matrix Multiplication( 矩阵压缩)

    Matrix Multiplication Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 18928   Accepted: ...

随机推荐

  1. SQL通用语法与分类

    通用语法 1.SQL可以单行或多行书写,以分号结尾. 2.MySQL数据库的SQL语句不区分大小写,关键字建议使用大写. 3.注释 1)单行注释: -- 注释内容  或 # 注释内容(MySQL特有) ...

  2. 暑集假训SCP提高拟模21

    \[だから妄想感傷代償連盟 \] \[愛を懐いて理想を叫んだ \] \[行き場のない愚者のメロディー \] \[再挑戦•転生•テレポーテーション \] \[何回だって 重ねて逝くんだ \] \[終わり ...

  3. schedtune.colocatte的作用

    schedtune.colocate 参数主要通过 /proc/sys/kernel/schedtune.colocate 接口进行配置.具体的使用方式和可选参数如下: 使用方法 你可以通过以下命令来 ...

  4. .NET 白板书写预测-曲线拟合

    白板软件书写速度是其最核心的功能,注册StylusPlugin从触摸线程拿触摸点数据并在另一UI线程绘制渲染是比较稳妥的方案,具体的可以查看小伙伴德熙的2019-1-28-WPF-高性能笔 - lin ...

  5. 深入探索Spring AI:源码分析流式回答

    在上一章节中,我们深入分析了Spring AI的阻塞式请求与响应机制,并探讨了如何增强其记忆能力.今天,我们将重点讲解流式响应的概念与实现.毕竟,AI的流式回答功能与其交互体验密切相关,是提升用户满意 ...

  6. 什么是 vite ?

    vite 是尤雨溪团队开发的新一代前端构建工具,作者尤雨溪宣传的时候表示 vite 是下一代的构建工具,并表示自己再也不会 webpack 了 : vite 的优点主要有3点 : 第一:开发环境中,无 ...

  7. Android复习(二)应用资源——>字符串

    转自:https://developer.android.google.cn/guide/topics/resources/string-resource#kotlin 字符串资源为您的应用提供具有可 ...

  8. 云原生周刊:Linkerd 发布 v2.14 | 2023.9.4

    开源项目推荐 Layerform Layerform 是一个 Terraform 包装器,可帮助工程师使用纯 Terraform 文件构建可重用的基础设施. 为了实现重用,Layerform 引入了层 ...

  9. 再见,Centos!

    近日,CentOS官方宣布CentOS系列稳定版Linux系统将停止维护,取而代之的是测试版的CentOS Stream,这也意味着CentOS将会退出历史舞台,因此引发了CentOS用户的强烈不满. ...

  10. Java编程案例(专题)

    文章目录 案例一:买飞机票 案例二:开发验证码 案例三:评委打分 案例四:数字加密 案例五:数组拷贝 案例六:抢红包 案例七:找素数 案例八:模拟双色球 8.1 手动投注 8.2 随机开奖号码 8.3 ...