变色龙之恋

在 JOI 动物园中,有着 \(2N\) 只变色龙,编号为 \(1\ldots 2N\)。其中,有 \(N\) 只变色龙的性别为 X,其余 \(N\) 只的性别为 Y。

每只变色龙都有一个原色。关于原色的可公开情报如下:

  • 所有性别为 X 的变色龙的原色不同。
  • 对于每只性别为 X 的变色龙,都存在唯一的原色与其相同的变色龙,且性别为 Y。

现在,JOI 动物园迎来了恋爱的季节。每只变色龙都爱上了另一只变色龙。关于恋爱对象的可公开情报如下:

  • 每只变色龙都很专一于唯一一只异性的变色龙。
  • 一只变色龙和它的恋爱对象的原色不同。
  • 不存在两只变色龙同时追求另一只变色龙。

你可以召集一部分变色龙来组织一场会议。对于一只参加会议的变色龙 \(s\),令 \(t\) 为它的恋爱对象。\(s\)肤色由以下方式决定:

  • 如果 \(t\) 参加了这场会议,则 \(s\) 的肤色为 \(t\) 的原色。
  • 如果 \(t\) 没参加这场会议,则 \(s\) 的肤色为 \(s\) 的原色。

一只变色龙的肤色可以在不同的会议间发生改变。对于你组织的一场会议,你可以得到场上所有变色龙的肤色的种类数。

由于变色龙也会感到厌烦,所以你最多只能组织 \(20\,000\) 场会议。同时你需要根据你得到的信息,确定有哪些变色龙的原色相同。
请你写一个程序在组织不超过 \(20\,000\) 场会议的前提下确定所有相同原色的变色龙。

\(2 \le N \le 500\)

题解

https://hk-cnyali.com/2020/03/24/「JOISC-2020-Day2」变色龙之恋-交互-二分图/
javascript:void(0)

先规定一些记号:

  1. \(x'\):与\(x\)同色的点。

  2. \(L(x)\)\(x\)喜欢的点。

  3. \(L^{-1}(x)\):喜欢\(x\)的点。

考虑先把所有大小为\(2\)的集合的答案问出来,不难发现答案只可能是\(1\)\(2\)

答案为\(1\)的三种情况:\(\{x,x'\},\{x,L(x)\},\{x,L^{-1}(x)\}\)。后两种会出现当且仅当\(L(x)\neq L^{-1}(x)\)

我们给答案为\(1\)的点对\((x, y)\)连一条边,那么每个点的度数只可能是\(1\)\(3\)

  • 若一个点度数为\(1\),则可以直接得到同色的点。

  • 若一个点度数为\(3\),设它为\(x\),三条边分别为\((x,x'),(x,L(x)),(x,L^{-1}(x))\)。只需询问\(\{x,x',L(x)\},\{x,x',L^{-1}(x)\},\{x,L(x),L^{-1}(x)\}\),这三个集合,答案分别为\(2,1,2\),那么我们就能得到\(L(x)\)\(L^{-1}(L(x))\)。最后我们简单处理一下就能得到同色点了。

这样就得到了询问\(O(n^2)\)的做法了。

考虑优化前半部分的复杂度。

不难发现这些边连成的图是二分图,设当前要处理\(i\)点与\(1 \sim i - 1\)点之间的连边,考虑将\(1 \sim i - 1\)这些点分成两个独立集,满足集合内部没有边(可以暴力染色处理)。

我们可以根据独立集的性质来判断是否有边: 一个集合\(S\)是独立集当且仅当\(\texttt{Query}(S) = |S|\)。也就是说,点\(i\)和一个独立集\(S\)有边当且仅当\(\texttt{Query}(\{S, i\}) \ne |S| + 1\)

于是可以通过二分来逐条找边了。复杂度\(O(n \log n)\)

CO int N=1e3+10;
vector<int> to[N];
int col[N];
vector<int> node[2];
int L[N],vis[N];

void dfs(int x,int c){
	col[x]=c,node[c].push_back(x);
	for(int y:to[x])if(col[y]==-1) dfs(y,!c);
}
void calc(vector<int> vec,int x){
	while(1){
		vec.push_back(x);
		if(Query(vec)==(int)vec.size()) return;
		vec.pop_back();
		int l=0,r=vec.size()-1,goal=-1;
		while(l<=r){
			int mid=(l+r)>>1;
			vector<int> tmp(vec.begin()+l,vec.begin()+mid+1);
			tmp.push_back(x);
			if(Query(tmp)!=(int)tmp.size()) goal=mid,r=mid-1;
			else l=mid+1;
		}
		if(goal==-1) return;
		int y=vec[goal];
		to[x].push_back(y),to[y].push_back(x);
		vec.erase(vec.begin()+goal);
	}
}
void get_edges(int x){
	node[0].clear(),node[1].clear();
	fill(col,col+x,-1);
	for(int i=1;i<x;++i)if(col[i]==-1) dfs(i,0);
	calc(node[0],x),calc(node[1],x);
}
void Solve(int n){
	for(int i=1;i<=2*n;++i) get_edges(i);
	for(int i=1;i<=2*n;++i)if(to[i].size()==3){
		for(int j=0;j<3;++j){
			vector<int> vec={i};
			for(int k=0;k<3;++k)if(k!=j)
				vec.push_back(to[i][k]);
			if(Query(vec)==1) {L[i]=to[i][j]; break;}
		}
	}
	for(int i=1;i<=2*n;++i){
		if(to[i].size()==1){
			if(!vis[i] and !vis[to[i][0]])
				Answer(i,to[i][0]),vis[i]=vis[to[i][0]]=1;
			continue;
		}
		for(int j=0;j<3;++j){
			if(L[i]==to[i][j] or L[to[i][j]]==i) continue;
			if(!vis[i] and !vis[to[i][j]])
				Answer(i,to[i][j]),vis[i]=vis[to[i][j]]=1;
		}
	}
}