不妨先从无解的情况下手,不难发现当 \(A \le B\) 时是一定无解的。

因为不诚实的 \(B\) 个人可以装作是诚实的,全部说自己这一方是诚实的对方是不诚实的我们就无法判断了。

下面我们就可以在 \(A > B\) 的情况下来处理这个交互问题了。

一个最基本最靠谱的方法是先找到一个诚实的人,然后用他把所有人全部问一遍。

因此下面我们的目的就是在 \(n\) 次询问以内找到一个诚实的人。

首先需要一个基于操作的观察:

若回答为 \(F\) 则两者之中必然存在一个人不诚实。

那么我们可以考虑使用这一条性质,将不诚实的人排除开来这样剩下的就是诚实的人了。

于此同时,因为我们只知道两者其一必不诚实,因此我们只能一次将两人一起排除。

并且,由于一次排除至多排除一个诚实的人,又 \(A > B\) 所以最终剩下的人一定是诚实的。

那么接下来的目的就是在 \(n\) 次内快速找到 \(B\) 组回答为 \(F\) 的人。

首先不难发现一个查询次数 \(n ^ 2\) 的暴力,每次取出一个没有被排除的人,用所有人询问他一次,正确性显然。

但是给予我们的询问上限是 \(n\) 次,但同时因为每个点至少要被询问一次,因为至少要保证每个不诚实的人都被至少查询一次(因为问不诚实的人的答案是不确定的)。

同时上限是 \(n\),所以我们只能让每个点被恰好被询问一次。

考虑用每个点恰好被询问一次来优化上面哪个暴力的做法。

首先还是先取出一个人来,用另一个人来询问他,如果询问结果为 \(F\) 那么把两人删去,继续递归这个操作;否则我们只能选择让这两个人都留下来,因为无法确定两者中是否存在不诚实的人,再往下递归。

同时因为每个点只能被查询一次,因此递归的时候只能让后面的人查询之前的查询者。

那么最终必然剩下的就是一条查询链,其中每个查询答案均为 \(T\)。

再来观察一下这条查询链,其中必然存在至少一个诚实的人,因为弹出次数最多为 \(B\) 显然 \(A > B\) 所以必然至少有一个诚实的人。

同时,你会发现这个查询链的最前端必然是诚实的人。

否则找到最靠前的哪个诚实的人,他必然能带走之前哪个不诚实的人并一起弹走。

这样我们就在 \(n\) 次查询内找到了一个诚实的人。

可以发现的是上面的这个做法本质上就是一个弹栈和入栈的过程,用栈实现即可。

复杂度 \(O(n)\)。

#include <bits/stdc++.h>
using namespace std;
#define rep(i, l, r) for (int i = l; i <= r; ++i)
const int N = 4000 + 5;
char s[5];
int n, A, B, top, st[N], ans[N];
int read() {
char c; int x = 0, f = 1;
c = getchar();
while (c > '9' || c < '0') { if(c == '-') f = -1; c = getchar();}
while (c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return x * f;
}
int main() {
A = read(), B = read(), n = A + B;
if(A <= B) { puts("Impossible"); return 0;}
rep(i, 1, n) {
if(!top) st[++top] = i;
else {
printf("? %d %d\n", i - 1, st[top] - 1), cout.flush();
scanf("%s", s + 1);
if(s[1] == 'Y') st[++top] = i;
else --top;
}
}
rep(i, 1, n) {
printf("? %d %d\n", st[1] - 1, i - 1), cout.flush();
scanf("%s", s + 1);
ans[i] = (s[1] == 'Y');
}
printf("! ");
rep(i, 1, n) printf("%d", ans[i]);
return 0;
}

AT2348 [ARC070D] HonestOrUnkind的更多相关文章

  1. AT2348 HonestOrUnkind

    传送门 显然\(a>b\)的情况下才有解 考虑先找出一个诚实的人,然后剩下的都可以在\(n\)次以内问出来了 发现如果一个人说另一个人是说谎的那么这两个人必有一个是说谎的,由于诚实的人严格多于不 ...

  2. [atARC070F]HonestOrUnkind

    考虑当$a\le b$时,构造两种方案,满足诚实的人不交,接下来要求对于任意询问,这两种方案的答案都有可能相同 考虑询问$(i,j)$,若$i$在两种方案中有一种不诚实,那么总可以让答案相同,又因为诚 ...

  3. AtCoder瞎做第二弹

    ARC 067 F - Yakiniku Restaurants 题意 \(n\) 家饭店,\(m\) 张餐票,第 \(i\) 家和第 \(i+1\) 家饭店之间的距离是 \(A_i\) ,在第 \( ...

  4. AtCoder刷题记录

    构造题都是神仙题 /kk ARC066C Addition and Subtraction Hard 首先要发现两个性质: 加号右边不会有括号:显然,有括号也可以被删去,答案不变. \(op_i\)和 ...

  5. 【AtCoder】ARC070

    ARC070 C - Go Home 题目大意:一只袋鼠第i秒可以向左或向右跳i步或者不跳,问从0跳到x的最小时间 就是1,2,3,4...k总和超过x的最小的k,因为如果超过了x的那部分需要减掉的那 ...

随机推荐

  1. Jenkins安装教程:Windows环境通过jenkins.war安装

    1.Windows操作系统下,安装jdk.tomcat.maven.git,并配置好对应的环境变量,安装教程请自行查询资料 2.将下载的jenkins.war放入到tomcat的webapp文件夹下, ...

  2. Kernel PCA and De-Noisingin Feature Spaces

    目录 引 主要内容 Kernel PCA and De-Noisingin Feature Spaces 引 kernel PCA通过\(k(x,y)\)隐式地将样本由输入空间映射到高维空间\(F\) ...

  3. opencv学习(六)——图像基本操作

    图像基本操作 一.访问和修改像素值 先来理解一下,图像与一般的矩阵或张量有何不同(不考虑图像的格式,元数据等信息).首先,一张图像有自己的属性,宽,高,通道数.其中宽和高是我们肉眼可见的属性,而通道数 ...

  4. SpringBoot集成MyBatis-Plus框架

    1.说明 本文介绍Spring Boot集成MyBatis-Plus框架, 重点介绍需要注意的地方, 是SpringBoot集成MyBatis-Plus框架详细方法 这篇文章的脱水版, 主要是三个步骤 ...

  5. Jenkins安装、配置与说明

    Jenkins是一个开源的.提供友好操作界面的持续集成(CI)工具,主要用于持续.自动的构建/测试软件项目.监控外部任务的运行. 这么解释很抽象,举个例子,我们开发完一个功能,我们要将项目发布打包好, ...

  6. 深入 Laravel 内核之IOC容器

    升级工厂前的准备工作 无规矩不成方圆,随着越来越多的行为出现,我们需要需要定下一些规范. 为了约束每一个行为的规范,需要定义一个行为接口: interface BehaviorInterface { ...

  7. SpringBoot 中拦截器的简介及使用方式

    拦截器简介 拦截器通常通过动态代理的方式来执行. 拦截器的生命周期由IoC容器管理,可以通过注入等方式来获取其他Bean的实例,使用更方便. 拦截器配置使用方式 实现拦截器接口: import jav ...

  8. Bom 基本使用以及定时器 倒计时案例

    BOM 是浏览器对象模型 它提供了独立内容而与浏览器窗口进行交互的对象,其核心对象是window 窗口加载事件 注意:window.onload 就可以吧JS代码写在页面元素的上方,因为onload是 ...

  9. CF149D游戏

    题目描述 Petya遇到了一个关于括号序列的问题: 给定一个字符串S,它代表着正确的括号序列,即("(")与 (")")是匹配的.例如:"(())() ...

  10. ConfigParser_读取配置文件信息

    ConfigParse简介 ConfigParser 在python中是用来解析配置文件的内置模块,直接导入使用 import configparser 使用该模块可以对配置文件进行增.读.改.删操作 ...