题意:交互题,交互库有长为$n$的$01$串$S$,你可以用字符串$T$询问$\sum\limits_{i=1}^n[S_i=T_i]$,要求用$1030$次询问问出$S$,$n=5000$

首先我们可以问出一个集合内有多少个$1$,接下来的询问就是指问集合内$1$的个数,记集合$S$的答案为$ans_S$

我们硬点一个方案的最后一次询问是问总共有多少$1$

如果可以用$k$次询问问出长为$n$的串,那么可以用$2k$次询问问出长为$2n+k-1$的串,构造如下

把串分成三部分:$1\cdots n,n+1\cdots 2n,2n+1\cdots 2n+k-1$,记为$A,B,C$

设对$A$询问的集合为$P_{1\cdots k}$,对$B$询问的集合为$Q_{1\cdots k}$

对$i=1\cdots k-1$,问$P_i\bigcup Q_i$和$(A-P_i)\bigcup Q_i\bigcup\{C_i\}$,倒数第二次问$A$

两个询问结果之和是$ans_A+2ans_{Q_i}+C_i$,和$ans_A$比较奇偶性可以知道$C_i$,继而推出$ans_{Q_i},ans_{P_i}$,做完之后也自然知道了$ans_{Q_k}$,于是递归下去做就行了

当$n=5000$时,这个做法需要$1024$次询问

感觉这个题很巧妙啊?

#include"game.h"
#include<string.h>
#include<vector>
using namespace std;
typedef vector<int> vi;
int n;
string t;
int sum;
int get(vi v){
	int siz;
	t=string(n,'1');
	siz=0;
	for(int x:v){
		if(x>n)break;
		t[x-1]='0';
		siz++;
	}
	return(siz-guess(t)+sum)/2;
}
vi al(int n){
	vi v;
	for(int i=1;i<=n;i++)v.push_back(i);
	return v;
}
vi sh(vi a,int b){
	for(int&x:a)x+=b;
	return a;
}
bool us[6010];
vi inv(vi a,int n){
	vi c;
	memset(us,0,n+1);
	for(int x:a)us[x]=1;
	for(int i=1;i<=n;i++){
		if(!us[i])c.push_back(i);
	}
	return c;
}
vi operator+(vi a,vi b){
	a.insert(a.end(),b.begin(),b.end());
	return a;
}
struct sol{
	int n;
	vector<vi>a;
	void add(vi v){
		a.push_back(v);
	}
}w[20];
void pre(){
	int n,k,i,j;
	vi t;
	w[1]={2,{{1},{1,2}}};
	for(i=2;i<=10;i++){
		k=w[i-1].a.size();
		n=w[i-1].n;
		w[i].n=n*2+k-1;
		for(j=0;j<k-1;j++){
			vi&u=w[i-1].a[j];
			w[i].add(u+sh(u,n));
			t=inv(u,n)+sh(u,n);
			t.push_back(n*2+j+1);
			w[i].add(t);
		}
		w[i].add(al(n));
		w[i].add(al(w[i].n));
	}
}
int res[6010];
void solve(int sh,int id,vi r){
	if(id==1){
		res[sh+1]=r[0];
		res[sh+2]=r[1]-r[0];
		return;
	}
	int n,k,i,c,t,al;
	vi ra,rb;
	n=w[id-1].n;
	k=w[id-1].a.size();
	c=r[k*2-2];
	al=r[k*2-1]-c;
	for(i=0;i<k-1;i++){
		t=res[sh+n*2+i+1]=((r[i*2]+r[i*2+1])^c)&1;
		al-=t;
		rb.push_back(t=(r[i*2]+r[i*2+1]-t-c)/2);
		ra.push_back(r[i*2]-t);
	}
	ra.push_back(c);
	rb.push_back(al);
	solve(sh,id-1,ra);
	solve(sh+n,id-1,rb);
}
const int mx=10;
string game(int _n,int k){
	n=_n;
	vi r;
	string s(n,'1');
	pre();
	sum=guess(s);
	for(vi v:w[mx].a)r.push_back(get(v));
	solve(0,mx,r);
	for(int i=0;i<n;i++)s[i]=res[i+1]+'0';
	return s;
}

最新文章

  1. Python OS模块常用函数说明
  2. HTTP 学习
  3. 【读书笔记】iOS-读取文本文件
  4. 【转】CSS清除浮动_清除float浮动
  5. android异步加载图片
  6. BZOJ_2194_快速傅立叶之二_(FFT+卷积)
  7. SharePoint 2010 master page 控件介绍(2):ribbon (一同事读听着像泪奔)
  8. [转]把项目从VS2005升级到VS2013
  9. 七内部排序算法汇总(插入排序、Shell排序、冒泡排序、请选择类别、、高速分拣合并排序、堆排序)
  10. UIImage 和 UIImageView区别
  11. access按钮事件在子窗体打开窗体或报表
  12. java.lang.SecurityException: Can&#39;t make field constructor accessible
  13. 2018-2019-2《Java程序设计》结对编程项目-四则运算 第一周 阶段性总结
  14. macaca 初试
  15. 在CentOS上配置SAMBA共享目录(转载)
  16. CodeForces - 95B(DFS)
  17. java学习笔记41(数据库连接池 C3p0连接池)
  18. 转:Vue2.0+组件库总结
  19. ACM:油田(Oil Deposits,UVa 572)
  20. javaweb下载文件

热门文章

  1. 前段性能----repaint和reflow
  2. Hive元数据配置到MySql
  3. docker 服务无法启动
  4. continue语句:编程把100-300之间的能被25整除的数输出
  5. Modular Production Line (MCMF)
  6. RPM包——查询
  7. luogu_3645: 雅加达的摩天楼
  8. [Python] Python忽略warning警告错误
  9. [HNOI2016]序列 CDQ+DP
  10. Node.js之判断字符串中是否包含某个字符串