题意:

有两个\(n \times m\)的矩阵\(A,B\),都是由\(1 \times 2\)的砖块铺成,代表初始状态和结束状态

有一种操作可以把两个砖块拼成的\(2 \times 2\)的矩形旋转\(90^{\circ}\)

问如何操作才能使初始状态转化为结束状态,无解输出\(-1\)

分析:

不妨假设\(m\)为偶数,否则可以旋转整个矩阵使得矩阵的列数为偶数

先找一个中间过度状态矩阵\(C\),它的每个砖块都是水平方向的

求出\(A \to C\)和\(B \to C\)的操作序列,因为操作是可逆的,所以就得到了\(A \to C \to B\)的操作序列

从第一行开始逐个扫描,遇到一个竖直方向的砖块就将它旋转,这是可能有两种情况:

  • 右边的砖块也是竖直方向的,那么可以直接旋转
  • 右边砖块是水平的,那么就递归到子问题:将右边的水平砖块先旋转过来,再一起旋转

算法的正确性不会证=_=

#include <cstdio>
#include <algorithm>
#include <vector>
#include <map>
using namespace std; const int maxn = 60;
typedef pair<int, int> PII;
#define ALL(x) (x).begin(), (x).end() int n, m, Max;
char s1[maxn][maxn], s2[maxn][maxn];
vector<PII> a, b; void rotate(char s[][maxn], int x, int y) {
if(s[x][y] == 'U') {
s[x][y] = s[x+1][y] = 'L';
s[x][y+1] = s[x+1][y+1] = 'R';
} else {
s[x][y] = s[x][y+1] = 'U';
s[x+1][y] = s[x+1][y+1] = 'D';
}
} bool check(char s[][maxn], int x, int y) {
if(s[x][y] == s[x][y+1] && s[x][y] == 'U' && s[x+1][y] == s[x+1][y+1] && s[x+1][y] == 'D') return true;
if(s[x][y] == s[x+1][y] && s[x][y] == 'L' && s[x][y+1] == s[x+1][y+1] && s[x][y+1] == 'R') return true;
return false;
} bool adjust(char s[][maxn], int x, int y, int flag, vector<PII>& a) {
if(x + 1 >= n || y + 1 >= m) return false;
if(check(s, x, y)) {
rotate(s, x, y);
a.emplace_back(x, y);
return true;
} else {
if(!adjust(s, x+1-flag, y+flag, flag^1, a)) return false;
rotate(s, x, y);
a.emplace_back(x, y);
return true;
}
} void op(char& c) {
switch(c) {
case 'L': c = 'U'; break;
case 'U': c = 'L'; break;
case 'R': c = 'D'; break;
case 'D': c = 'R'; break;
}
} void change(char s[][maxn]) {
for(int i = 0; i < n; i++)
for(int j = 0; j < m; j++) op(s[i][j]);
for(int i = 0; i < Max; i++)
for(int j = 0; j < i; j++)
swap(s[i][j], s[j][i]);
} int main()
{
scanf("%d%d", &n, &m);
for(int i = 0; i < n; i++) scanf("%s", s1[i]);
for(int i = 0; i < n; i++) scanf("%s", s2[i]);
Max = n > m ? n : m;
bool changed = false;
if(m & 1) { change(s1); change(s2); swap(n, m); changed = true; } for(int i = 0; i < n; i++) {
for(int j = 0; j < m; j += 2) {
if(s1[i][j] != 'L') {
if(!adjust(s1, i, j, 1, a)) { puts("-1"); return 0; }
}
if(s2[i][j] != 'L') {
if(!adjust(s2, i, j, 1, b)) { puts("-1"); return 0; }
}
}
} reverse(ALL(b));
a.insert(a.end(), ALL(b));
printf("%d\n", (int)a.size());
for(pair<int, int> t : a) {
if(changed) swap(t.first, t.second);
printf("%d %d\n", t.first + 1, t.second + 1);
} return 0;
}

CodeForces 778D Parquet Re-laying 构造的更多相关文章

  1. CF 778D Parquet Re-laying——构造

    题目:http://codeforces.com/problemset/problem/778/D 完全没思路……就看了题解. 很好地思路是考虑操作可逆,所以起始状态和最终状态都变到一个中转状态,即都 ...

  2. Educational Codeforces Round 10 B. z-sort 构造

    B. z-sort 题目连接: http://www.codeforces.com/contest/652/problem/B Description A student of z-school fo ...

  3. Codeforces 707C Pythagorean Triples(构造三条边都为整数的直角三角形)

    题目链接:http://codeforces.com/contest/707/problem/C 题目大意:给你一条边,问你能否构造一个包含这条边的直角三角形且该直角三角形三条边都为整数,能则输出另外 ...

  4. Codeforces 1246D/1225F Tree Factory (构造)

    题目链接 https://codeforces.com/contest/1246/problem/D 题解 首先考虑答案的下界是\(n-1-dep\) (\(dep\)为树的深度,即任何点到根的最大边 ...

  5. Codeforces - 1202D - Print a 1337-string... - 构造

    https://codeforces.com/contest/1202/problem/D 当时想的构造是中间两个3,然后前后的1和7组合出n,问题就是n假如是有一个比较大的质数因子或者它本身就是质数 ...

  6. Codeforces 743C - Vladik and fractions (构造)

    Codeforces Round #384 (Div. 2) 题目链接:Vladik and fractions Vladik and Chloe decided to determine who o ...

  7. Codeforces 1368E - Ski Accidents(构造+思维)

    Codeforces 题面传送门 & 洛谷题面传送门 神仙构造题(不过可能我构造太烂了?) 首先考虑这个奇奇怪怪的 \(\dfrac{4}{7}\),以及这个每个点出度最多为 \(2\) 的条 ...

  8. Codeforces 1270E - Divide Points(构造+奇偶性)

    Codeforces 题目传送门 & 洛谷题目传送门 显然,直接暴力枚举是不可能的. 考虑将点按横纵坐标奇偶性分组,记 \(S_{i,j}=\{t|x_t\equiv i\pmod{2},y_ ...

  9. codeforces 622C. Optimal Number Permutation 构造

    题目链接 假设始终可以找到一种状态使得值为0, 那么两个1之间需要隔n-2个数, 两个2之间需要隔n-3个数, 两个3之间隔n-4个数. 我们发现两个三可以放到两个1之间, 同理两个5放到两个3之间. ...

随机推荐

  1. Azure CDN:氮气加速已开启,司机们请做好准备

    在上一周,我们向各位小伙伴介绍了通过 Azure CDN 高级版服务为 HTTPS 应用加速的做法,漏掉的小伙伴可以点击这里穿越回去补课哦.那我们今天讲点什么呢?当然是 CDN 最重要的价值:改善应用 ...

  2. Linux CentOS下部署Java Web项目

    本文讲解如何在Linux CentOS下部署Java Web项目的步骤. 一.环境准备: (1)Linux CentOS (2)apache-tomcat-9.0.10 (3)XShell 二.启动t ...

  3. 04、Spark Standalone集群搭建

    04.Spark Standalone集群搭建 4.1 集群概述 独立模式是Spark集群模式之一,需要在多台节点上安装spark软件包,并分别启动master节点和worker节点.master节点 ...

  4. 使用QJM实现HDFS的HA配置

    使用QJM实现HDFS的HA配置 1.背景 hadoop 2.0.0之前,namenode存在单点故障问题(SPOF,single point of failure),如果主机或进程不可用时,整个集群 ...

  5. ABI与编译器:ABI是由内核和工具链定义和实现的

    http://book.51cto.com/art/201412/460857.htm <Linux系统编程(第2版)>第1章入门和基本概念,这一章着眼于Linux系统编程的基础概念并从程 ...

  6. 阿里云主机ss

    https://promotion.aliyun.com/ntms/act/vm/aliyun-group/buy.html?group=HdcwGIaf6i

  7. 如何迅速掌握并提高linux运维技能(收藏文)

    如何迅速掌握并提高linux运维技能   文章来源于南非蚂蚁   之前曾经写过一篇如何学习Linux的文章,得到了很多反馈,大家都在分享自己的学习经验和体会,并且也提出了不少意见和建议.学习这个事情其 ...

  8. veritas.com常用资源汇总

    NetBackup 8.1.2文档(合集) https://www.veritas.com/support/en_US/article.100044086   NetBackup产品组停止支持生命周期 ...

  9. iOS 有些库只能在真机上运行,不能在模拟器上运行的解决方式

    在开发中,多少肯定会用到第三方的东西,或许大家也和我一样遇到到这样的情况,有些库正好适合自己的需求,但是这个库却只支持真机上运行,在模拟器上编译却不通过, 一般情况下,.a静态包,你刚刚导入的时候,不 ...

  10. Hive 常用命令和语句

    示例数据库为 db_hive 1. 创建表 create-table.sql create table if not exists db_hive.tb_user ( id int, username ...