AT2348 [ARC070D] HonestOrUnkind
不妨先从无解的情况下手,不难发现当 \(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的更多相关文章
- AT2348 HonestOrUnkind
传送门 显然\(a>b\)的情况下才有解 考虑先找出一个诚实的人,然后剩下的都可以在\(n\)次以内问出来了 发现如果一个人说另一个人是说谎的那么这两个人必有一个是说谎的,由于诚实的人严格多于不 ...
- [atARC070F]HonestOrUnkind
考虑当$a\le b$时,构造两种方案,满足诚实的人不交,接下来要求对于任意询问,这两种方案的答案都有可能相同 考虑询问$(i,j)$,若$i$在两种方案中有一种不诚实,那么总可以让答案相同,又因为诚 ...
- AtCoder瞎做第二弹
ARC 067 F - Yakiniku Restaurants 题意 \(n\) 家饭店,\(m\) 张餐票,第 \(i\) 家和第 \(i+1\) 家饭店之间的距离是 \(A_i\) ,在第 \( ...
- AtCoder刷题记录
构造题都是神仙题 /kk ARC066C Addition and Subtraction Hard 首先要发现两个性质: 加号右边不会有括号:显然,有括号也可以被删去,答案不变. \(op_i\)和 ...
- 【AtCoder】ARC070
ARC070 C - Go Home 题目大意:一只袋鼠第i秒可以向左或向右跳i步或者不跳,问从0跳到x的最小时间 就是1,2,3,4...k总和超过x的最小的k,因为如果超过了x的那部分需要减掉的那 ...
随机推荐
- Oracle 创建数据表以及对数据表、字段、主外键、约束的操作
选择主键的原则: 最少性 尽量选择使用单个键作为主键 稳定性 尽量选择数值更新少的列作为主键 1.创建数据表(CREATE TABLE) --创建数据表Student create table Stu ...
- Log4j2进阶使用(按大小时间备份日志)
1.进阶说明 本文介绍Log4j2进阶使用, 基本使用请参考Log4j2基本使用入门. 本文基于上面的基本使用入门, 主要介绍按照日志大小和时间备份日志, 并且限制备份日志的个数, 以及删除过期的备份 ...
- MySQL支持IPv6
开启和验证MySQL支持IPv6的方法, 此处使用的MySQL版本为mysql-5.5.35-linux2.6-x86_64. 1.验证操作系统支持IPv6,此处是Linux操作系统 ping6 :: ...
- vue 多级路由嵌套后打开页面是空白
在多层路由嵌套时,一级子目录必须有一个页面并且添加一具<router-view>,否则路由跳转页面为空,没有任何显示 来自为知笔记(Wiz)
- Nginx 防爬虫设置
在conf下 vi 一个文件agent_deny.conf 添加如下内容 #禁止Scrapy|curl等工具的抓取 if ($http_user_agent ~* (Scrapy|Curl|Http ...
- js- float类型相减 出现无限小数的问题
6.3 -1.1 是不是应该等于5.2? 但是js 会导致得出 5.19999999999的结果 怎么办?可以先先乘100 后相减,然是用方法 舍入为最接近的整数,然后再除于100, Math.rou ...
- Leetcode算法系列(链表)之删除链表倒数第N个节点
Leetcode算法系列(链表)之删除链表倒数第N个节点 难度:中等给定一个链表,删除链表的倒数第 n 个节点,并且返回链表的头结点.示例:给定一个链表: 1->2->3->4-&g ...
- HTML5基本结构和语法
1.1HTML5文档基本结构 HTML5文档省略了<html>,<head>,<body>等元素,使用HTML5的DOCTYRE声明文档类型,简化<meta& ...
- 使用Python 爬取 京东 ,淘宝。 商品详情页的数据。(避开了反爬虫机制)
以下是爬取京东商品详情的Python3代码,以excel存放链接的方式批量爬取.excel如下 代码如下 from selenium import webdriver from lxml import ...
- 《剑指offer》面试题18. 删除链表的节点
问题描述 给定单向链表的头指针和一个要删除的节点的值,定义一个函数删除该节点. 返回删除后的链表的头节点. 注意:此题对比原题有改动 示例 1: 输入: head = [4,5,1,9], val = ...