题意:

有两个\(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. SharePoint 栏的三种名字Filed :StaticName、 InternalName、 DisplayName

    SharePoint 的栏,有3个名字, StaticName InternalName  DisplayName. 当在第一次创建栏的时候,这3个名字一起进行创建,并且都一样. <FIELD  ...

  2. 我的ORM框架

    任何系统的基础,都可以算是各种数据的增删改查(CRUD).最早操作数据是直接在代码里写SQL语句,后来出现了各种ORM框架.C#下的ORM框架有很多,如微软自己的Entity Framework.第三 ...

  3. Thymeleaf 随记

    一.基础写法: th:text='${数据}  ,其中text可以修改成其他,如href,value,class....看需求 <p th:text='${后台返回的数据}'>静态文本&l ...

  4. 如何查看某个用户指定时间段的ABAP开发记录

    输入用户名和想查询的时间段: 执行得到结果.双击可查看具体代码: 工具源代码: REPORT tool_dev_history. PARAMETERS: name TYPE usr02-bname O ...

  5. Linux Mint,Ubuntu 18 ,Deepin15.7 安装mysql 没有提示输入密码,修改root用户密码过程

    刚刚装Deepin15.7 和 MySQL5.7 发现没有提示用户输入密码的过程(近日发现Linux Mint 和 Ubuntu18 也适用) 百度了一大堆如何修改root密码 也没什么卵用,终于这篇 ...

  6. 解决robotframework安装时提示wxPython not found问题

    背景:想把现在pc的项目做成关键字的形式,可以让功能测试人员也参与到自动化测试中,于是就找到robotframework这个框架,试用下怎么样,在安装时就遇到很多问题,安装的帖子有很多,很详细,如:h ...

  7. 关于Content-Type: multipart/form-data的支持

    关于Content-Type: multipart/form-data的支持  发布于 537天前  作者 yzhang226  303 次浏览  复制  上一个帖子  下一个帖子  标签: 无 今天 ...

  8. 百度非会员满速下载利器(IDM)Internet Download Manager v6.30.8 中文特别版

    下载利器(IDM)Internet Download Manager v6.30.8 中文特别版 所属分类:工具软件 应用平台:Windows 资源版本:v6.30.8 最后更新:2018年04月14 ...

  9. Matlab将矩阵保存为图像

    imwrite(image,'image.jpg'); image为矩阵的内容 image.jpg为要保存的图像的名字

  10. openstack cinder api对应的命令行接口

    a) Create Volume         cinder create 1 --display-name admin-volume1     cinder create --display-na ...