@description@

这是一道交互题。

现在有一个 n*n 的矩阵,每个位置是 0 或 1。现在已知 n 为奇数、左上角为 1、右下角为 0。

你可以向交互库给出询问 "? x1 y1 x2 y2",交互库会回答是否存在一条路径从 (x1, y1) 出发仅往右或往下走到达 (x2, y2),且路径上所有数拼起来形成回文串。如果存在为 1,否则为 0。

请注意询问时必须保证 x1 <= x2, y1 <= y2 且 (x2 - x1) + (y2 - y1) > 1,即 (x1, y1) 在 (x2, y2) 的左上方,且它们不能相邻。

请使用不超过 n^2 组询问,还原出这个矩阵。

Input

第一行包含一个奇数 n (3 <= n < 50)。

Interaction

你可以按照格式 "? x1 y1 x2 y2" 给出询问,含义如上。

当你决定好了矩阵中的所有元素,请输出一行 "!",紧接着输出你的矩阵。

Example

Input

3

0

1

0

1

1

1

1

Output

? 1 1 1 3

? 1 1 2 3

? 2 1 2 3

? 3 1 3 3

? 2 2 3 3

? 1 2 3 2

? 1 2 3 3

!

100

001

000

@solution@

考虑一个长度为 3 的询问,实际上 是不是回文串 只跟 开头与末尾是否相同 有关。

于是将矩阵黑白染色后,可以通过若干长度为 3 的询问得到黑格之间的相对关系,白格之间的相对关系。这个可以用带权并查集。

因为是相对关系,所以 n^2 个点只询问 n^2 - 1 次询问,实际上因为已经给定左上与右下,所以实际还会少一点点。

我们假设左上角为黑格,则所有黑格的数字都已经可以确定。

我们接下来就是尝试找到 1 个询问,使我们可以确定某个白格的数字,这样总询问次数也不会超过上界。

考虑一个 2*3 的子矩形,询问左上角的 (x, y) 到右下角的 (x + 1, y + 2)(同理可以推到 3*2 上面去)。

因为是 2*3,所以 (x, y) 与 (x + 1, y + 2) 必然有一个黑格,不失一般性设 (x, y) 为黑格。

假如 (x + 1, y + 2) 填充与 (x, y) 不同,必然没有回文串;否则假如填充相同时一定存在回文串,我们就可以通过这个询问确定白格。

讨论几种情况:

(1)假如 (x + 1, y + 1) ≠ (x, y + 2),则路径 (x, y) -> (x, y + 1) -> (x, y + 2) -> (x + 1, y + 2) 与 (x, y) -> (x, y + 1) -> (x + 1, y + 1) -> (x + 1, y + 2) 中必然有一个回文串。

(2)假如 (x + 1, y) ≠ (x, y + 1),则同理可得路径 (x, y) -> (x + 1, y) -> (x + 1, y + 1) -> (x + 1, y + 2) 与 (x, y) -> (x, y + 1) -> (x + 1, y + 1) -> (x + 1, y + 2) 中必然有一个回文串。

(3)假如上述两种情况对于任意子矩形都不满足,等价于所有 左上 - 右下 的对角线上的数全部相同。

这时候考虑,假如 (x, y) = (x + 1, y + 1) 与 (x + 1, y + 2) = (x, y + 1) 同时满足或者同时不满足,则也必然存在一个回文串。

否则当 (x + 1, y + 2) ≠ (x, y + 1) 时,(x, y) 所在对角线与 (x + 1, y + 1) 所在对角线相同;又因以 (x, y + 1) 为左上角时,又可以推出 (x + 1, y + 1) 与 (x + 2, y + 2) 所在对角线相同。进一步推出所有黑格对角线相同,然而这与左上角为 1 右下角为 0 矛盾。

当 (x + 1, y + 2) = (x, y + 1),大概就是所有黑格对角线形成 10101... 这样间隔开来的形式。然而因为 n 为奇数,所以有奇数条黑格对角线,于是这又与左上角为 1 右下角为 0 矛盾。

综上,以上 3 种情况必然有一个成立,这个算法可以找出解。

@accepted code@

#include<cstdio>
int query(int x1, int y1, int x2, int y2) {
printf("? %d %d %d %d\n", x1, y1, x2, y2);
fflush(stdout);
int x; scanf("%d", &x);
return x;
}
int num[50 + 5][50 + 5], n;
int fa[50*50 + 5], f[50*50 + 5];
int find(int x) {
if( fa[x] == x ) return x;
int y = fa[x]; fa[x] = find(fa[x]);
f[x] ^= f[y];
return fa[x];
}
void link(int x, int y, int k) {
int fx = find(x), fy = find(y);
fa[fx] = fy, f[fx] = f[x]^f[y]^k;
}
int d[50*50 + 5];
int main() {
scanf("%d", &n);
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
num[i][j] = (i - 1)*n + j;
for(int i=1;i<=n*n;i++)
fa[i] = i, f[i] = 0;
link(num[1][1], num[2][2], !query(1, 1, 2, 2));
for(int i=3;i<=n;i+=2)
link(num[1][i], num[1][i-2], !query(1, i-2, 1, i));
for(int i=4;i<=n;i+=2)
link(num[2][i], num[2][i-2], !query(2, i-2, 2, i));
for(int i=4;i<=n;i+=2)
for(int j=2;j<=n;j+=2)
link(num[i-2][j], num[i][j], !query(i-2, j, i, j));
for(int j=1;j<=n;j+=2)
for(int i=3;i<=n;i+=2)
link(num[i-2][j], num[i][j], !query(i-2, j, i, j));
link(num[2][1], num[3][2], !query(2, 1, 3, 2));
for(int i=3;i<=n;i+=2) {
link(num[i-2][2], num[i][2], !query(i-2, 2, i, 2));
link(num[2][i-2], num[2][i], !query(2, i-2, 2, i));
}
for(int i=4;i<=n;i+=2)
for(int j=1;j<=n;j+=2) {
link(num[i-2][j], num[i][j], !query(i-2, j, i, j));
link(num[j][i-2], num[j][i], !query(j, i-2, j, i));
}
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
find(num[i][j]);
for(int i=1;i+2<=n;i++) {
for(int j=1;j+1<=n;j++) {
if( (f[num[i][j]] == f[num[i+2][j]]) == (f[num[i+1][j]] == f[num[i+2][j+1]]) || f[num[i+1][j]] != f[num[i][j+1]] || f[num[i+2][j]] != f[num[i+1][j+1]] ) {
if( (i+j) & 1 ) d[fa[num[i][j]]] = f[num[i+2][j+1]]^(!query(i, j, i+2, j+1))^f[num[i][j]];
else d[fa[num[i+2][j+1]]] = f[num[i][j]]^(!query(i, j, i+2, j+1))^f[num[i+2][j+1]];
puts("!");
for(int i=1;i<=n;i++) {
for(int j=1;j<=n;j++)
printf("%d", d[fa[num[i][j]]]^f[num[i][j]]);
puts("");
}
return 0;
}
if( (f[num[j][i]] == f[num[j][i+2]]) == (f[num[j][i+1]] == f[num[j+1][i+2]]) || f[num[j][i+1]] != f[num[j+1][i]] || f[num[j][i+2]] != f[num[j+1][i+1]] ) {
if( (i+j) & 1 )
d[fa[num[j][i]]] = f[num[j+1][i+2]]^(!query(j, i, j+1, i+2))^f[num[j][i]];
else
d[fa[num[j+1][i+2]]] = f[num[j][i]]^(!query(j, i, j+1, i+2))^f[num[j+1][i+2]];
puts("!");
for(int i=1;i<=n;i++) {
for(int j=1;j<=n;j++)
printf("%d", d[fa[num[i][j]]]^f[num[i][j]]);
puts("");
}
return 0;
}
}
}
}

@details@

因为时间不够我也没仔细研究官方给出的构造方法,不过构造方法据我了解也比较多。

像这种构造题主要还是从小的数据范围考虑起,把复杂问题简单化吧。

@codeforces - 1205C@ Palindromic Paths的更多相关文章

  1. Codeforces 1205C Palindromic Paths (交互题、DP)

    题目链接 http://codeforces.com/contest/1205/problem/C 题解 菜鸡永远做着变巨的梦 然而依然连div1BC题都不会做 要是那天去打cf怕是又要1题滚粗了.. ...

  2. Educational Codeforces Round 89 (Rated for Div. 2) C Palindromic Paths

    题目链接:Palindromic Paths 题意: 给你一个n行m列的矩阵,这个矩阵被0或者1所填充,你需要从点(1,1)走到点(n,m).这个时候会有很多路径,每一条路径对应一个01串,你可以改变 ...

  3. [USACO15OPEN]回文的路径Palindromic Paths

    [USACO15OPEN]回文的路径Palindromic Paths 题目描述 Farmer John's farm is in the shape of an N \times NN×N grid ...

  4. TOJ 5020: Palindromic Paths

    5020: Palindromic Paths  Time Limit(Common/Java):10000MS/30000MS     Memory Limit:65536KByteTotal Su ...

  5. 题解 P3126 【[USACO15OPEN]回文的路径Palindromic Paths】

    P3126 [USACO15OPEN]回文的路径Palindromic Paths 看到这题题解不多,蒟蒻便想更加通俗易懂地分享一点自己的心得,欢迎大佬批评指正^_^ 像这种棋盘形的两边同时做的dp还 ...

  6. CodeForces 835D - Palindromic characteristics | Codeforces Round #427 (Div. 2)

    证明在Tutorial的评论版里 /* CodeForces 835D - Palindromic characteristics [ 分析,DP ] | Codeforces Round #427 ...

  7. Educational Codeforces Round 89 (Rated for Div. 2) C. Palindromic Paths(贪心)

    题目链接:https://codeforces.com/contest/1366/problem/C 题意 有一个 $n \times m$ 的 $01$迷宫,要使从 $(1,1)$ 到 $(n,m) ...

  8. Educational Codeforces Round 89 (Rated for Div. 2) C. Palindromic Paths (思维)

    题意:有一个\(n\)x\(m\)的矩阵,从\((1,1)\)出发走到\((n,m)\),问最少修改多少个数,使得所有路径上的数对应相等(e.g:\((1,2)\)和\((n-1,m)\)或\((2, ...

  9. [USACO15OPEN]回文的路径Palindromic Paths 2.0版

    题目描述 农夫FJ的农场是一个N*N的正方形矩阵(2\le N\le 5002≤N≤500),每一块用一个字母作标记.比如说: ABCD BXZX CDXB WCBA 某一天,FJ从农场的左上角走到右 ...

随机推荐

  1. DjangoORM查询、分页、ckeditor

    查询数据 Django的批量查询(查询所有,或者条件查询)返回的是queryset对象. Queryset对象是一个惰性对象,在不执行 1.排序 2.循环 3.截取 操作的情况下,不会遍历序列的内容. ...

  2. 洛谷P1969 [NOIP2013提高组Day2T1] 积木大赛

    P1969 积木大赛 题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前, ...

  3. 【每日一linux命令7】用户及用户组

    一.查询用户及用户组相关命令 1.whoami 查询当前登录的用户名 2.groups 查询当前登录用户名所在的用户组 3.groups root 查询root用户名所在的用户组 二.怎么批量查看用户 ...

  4. The method getTextContent() is undefined ?

    晚上下班的时候,把班上写了半截的代码带了回来.结果回到家后出乎意料的是回来的时候将代码导入eclipse后,下面这行代码就直接报错了,显示 getTextContent()未定义 . ((Elemen ...

  5. hive-oracle-常用分析函数-窗口函数

    常用的分析函数如下所列: row_number() over(partition by ... order by ...)rank() over(partition by ... order by . ...

  6. FastAdmin 使用 phpmail 出现 spl_autoload_register 错误

    FastAdmin 使用 phpmail 出现 spl_autoload_register 错误 现象 意思是 __autoload() 已经废弃 问题来源于:https://ask.fastadmi ...

  7. FastAdmin 安装后点登录没有反应怎么办?

    FastAdmin 安装后点登录没有反应怎么办? 很多小伙伴安装后点"登录"没有反应. 这个 URL 是对的,但是页面就是不改变. 如果这时候点击 URL 变了,那没有到登陆界面, ...

  8. SDUT-3376_数据结构实验之查找四:二分查找

    数据结构实验之查找四:二分查找 Time Limit: 30 ms Memory Limit: 65536 KiB Problem Description 在一个给定的无重复元素的递增序列里,查找与给 ...

  9. 洛谷 P1948 [USACO08JAN]电话线Telephone Lines 最短路+二分答案

    目录 题面 题目链接 题目描述 输入输出格式 输入格式 输出格式 输入输出样例 输入样例 输出样例 说明 思路 AC代码 题面 题目链接 P1948 [USACO08JAN]电话线Telephone ...

  10. 如何用最暴力的方法改写Liferay的原生portlet

    最近在论坛上看到有人问如何改写Liferay原有的calendar portlet. 然后研究了一下,直接从portal中把calendar portlet的源码拷贝出来,然后修改再部署上去,但是这个 ...