首页 > 其他分享 >【题解】CF1854D Michael and Hotel

【题解】CF1854D Michael and Hotel

时间:2023-09-08 19:44:53浏览次数:45  
标签:环上 int 题解 Hotel Michael mid 环长 找到 我们

交互题。

考虑题意即为找到 \(1\) 所在内向基环树上的所有点。

我们考虑我们怎么找到环上的点,我们考虑我们可以 \(O(\log n)\) 询问到一个环上的点,方法即为将 \(k\) 定为一个大数,然后二分点集。然后我们便可以在 \(O(n\log n)\) 的时间复杂度内找到所有环上的点(我们一会儿再讲怎样优化)。

我们找到环上的点了之后,我们再将 \(S\) 设为环上的点,\(k\) 设为一个极大数,\(u\) 从 \(1\sim n\) 遍历,如果返回 \(1\) 即为可以到达,否则不能到达,这样我们就可以在 \(O(n)\) 的时间内找到所有链上的点。

我们显然现在需要将找环上的点的操作再精简一下。

我们考虑我们找链上的点时的方法十分地高效,我们可以略作修改来找环。

我们考虑假设我们找到了环上的长度为 \(len\) 的链,那么我们可以将 \(S\) 设为已经找到的环上的点,\(k = len\),\(u\) 从 \(1\sim n\) 遍历。返回 \(1\) 便将点加入环,这样,我们每次的环长就可以倍增了,停止条件就是环长没有倍增(即绕了一圈回来了)。

上面的方法显然可能将环外的点也加进来,但是显然没有任何印象。

我们便可以将两种找环上点的方法结合起来,先每次 \(\log n\) 询问环上的单点0,然后每次 \(O(n)\) 倍增环上的点。由下图可知,一开始询问出长度为 \(56\) 链是最优的,当然上下浮动一点点也无伤大雅。

image

code:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int NN = 5e2 + 8;

int n,tot,rt = 1;

bool check(int tim,int l,int r){
	printf("? %d %d %d ",rt,tim,r - l + 1);
	for(int i = l; i <= r; ++i) printf("%d ",i);
	puts("");fflush(stdout);
	
	int op;scanf("%d",&op);
	return op;
}

set<int> Nodes;
bool vis[NN];

void Get_Part_Of_Ring(){
	int l = 1,r = n;
	while(l < r){
		int mid = (l + r) / 2;
		if(check(1000,l,mid)) r = mid;
		else l = mid + 1;
	}
	Nodes.insert(l);
	rt = l;
	for(int i = 1; i <= 62; ++i){
		l = 1; r = n;
		while(l < r){
			int mid = (l + r) / 2;
			if(check(1,l,mid)) r = mid;
			else l = mid + 1;
		}
		if(vis[l]) break;
		vis[l] = 1;
		Nodes.insert(l);
		rt = l;
	}
}
void Get_Ring(){//事实上应该包括了一部分的链 
	if(Nodes.size() != 63) return;//环找完了,在上一个函数里面  
    int sz = Nodes.size(); 
    while(1){
        tot = 0;
        for(int i = 1; i <= n; ++i) if(!vis[i]){
        	printf("? %d %d %d ",i,sz,Nodes.size());
            for(auto j : Nodes) printf("%d ",j);
            puts("");fflush(stdout);
            
            int op;scanf("%d",&op);
            if(op) vis[i] = 1,Nodes.insert(i);
        }//将环扩倍,当然也会有环外面链的部分被包含进来,但是无伤大雅 
        sz *= 2;
        if(sz > Nodes.size()) break;//有一部分重复了,环长没有扩倍,说明环找完了 
    }
}
void Get_Link(){
	for(int i = 1;i <= n;i++)if(!vis[i]){
        printf("? %d %d %d ",i,1000,Nodes.size());;
        for(auto j : Nodes) printf("%d ",j);
        puts("");fflush(stdout);
        
        int op;scanf("%d",&op);
        if(op == 1) Nodes.insert(i);
    }
}
void print(){
	printf("! %d ",Nodes.size());
	for(auto i : Nodes) printf("%d ",i);
    puts("");fflush(stdout);
}
int main(){
	scanf("%d",&n);
	Get_Part_Of_Ring();
	Get_Ring();
    Get_Link();
    print();
    return 0;
}

标签:环上,int,题解,Hotel,Michael,mid,环长,找到,我们
From: https://www.cnblogs.com/rickylin/p/17688420.html

相关文章

  • 【题解】CF1854C Expected Destruction
    你考虑,我们如果没有重合就将元素删去的操作,我们就有答案:\(n\times(m+1)-\sum\limits_{i=1}^na_i\)但是,我们显然最后的答案是小于这个的,如果有两个数在\(i\)相撞,那么我们的答案就会减少\((m-i+1)\)我们设\(f_{i,j}\)表示两个数分别在\(i\)和\(j\)的概率\((i\leqj......
  • 【疑难解决】运行EasyRTSPSever组件提示程序无法启动问题解决
    RTSP协议以客户服务器方式工作,它是一个多媒体播放控制协议,用来使用户在播放从因特网下载的实时数据时能够进行控制,如:暂停/继续、后退、前进等。因此RTSP又称为“因特网录像机遥控协议”。我们的RTSP-Sever组件EasyRTSPSever就是一款比较便捷的组件。我们有开发者在测试EasyRTSPS......
  • live555作为RTSP流媒体服务器时Server端多track而客户端仅请求一个track,当客户端关闭
    当我们使用live555作为流媒体服务器时,某个通道对应的所有客户端断开后,不能正常回调关闭。某一通道同时支持视频和音频输出,即video和audio两个trackVLC和EasyPlayer播放库来中的RTSPClient则都会请求(所以不存在问题);而某些客户端则只请求了一个track,比如video;此时再关闭......
  • live555做流媒体服务器时解决rtp over udp模式下, 客户端没有发送teardown时直接关闭
    在我们使用live555作为RTSP服务器时,客户端在rtpoverudp模式下,rtsp客户端没有发送teardown而直接断开连接时需要等待65秒才回调关闭的问题。分析问题在RTSPClientConnection中没有保存相应的session值,所以在RTSPClientConnection断开时,并没有删除相应的RTSPClientSession;解......
  • 【题解】CF1854B Earn or Unlock
    你考虑,我们很容易地可以构造一个\(n^2\)的DP:\(f_{i,j}\)表示当前在\(i\)张牌,还可以摸\(j\)张牌的最大分数。转移也很好转移,你考虑一眼就会。但是我们显然要缩减复杂度,我们看到数据范围\(10^5\),想到了根号。分块???显然不行。莫队???都没有区间查询,怎么行呢?然后你苦思冥想......
  • 【题解】AtCoder Regular Contest 161
    评价:感觉这场题目质量不咋地啊,都是一些乱搞题A.MakeM题目描述:\(N\)是一个正奇数。我们称一个长度为\(N\)的序列\(S\)是M型序列,当前仅当对于所有的\(i=2,4,6,\dots,N-1\)(即偶数位),都有\(S_{i-1}<S_{i}\)且\(S_{i}>S_{i+1}\)。现在给定你一个长度为\(N\)的序列\(A......
  • CF613D 题解
    一、题目描述:给你一颗$n$个点的树,有$m$组询问。一个点如果被攻占,那么这个点就不能通行了。第$i$次询问给出$k_i$个关键点,关键点不能被攻占。求最少攻占多少个点可以使得关键点两两不连通。若不可能,输出$-1$。数据范围:$1\len,m\le1\times10^5......
  • 9月杂题题解
    arc124_e一种方案的权值为\(\prod\limits_{1\leqi\leqn}b_i\),考虑其组合意义,就是每个人在自己最终的球中选一个。可以发现要么拿自己原来的球,要么拿上一个人传来的球。定义状态:\(f(i,0)\)为第\(i\)个人拿自己的球,考虑前\(i-1\)个人的答案。\(f(i,1)\)为第\(i\)个......
  • nvm有下载版本,切换版本成功,node -v还是切换前的版本问题解决
    是因为在下载nvm之前,电脑中的node版本已经存在了,所以需要将之前的node版本全部清楚干净!卸载node之前请node-v查看一下现在的版本,记住这个版本,切记切记!!!!!控制面板中卸载node.;卸载已安装过的NVM;没装过NVM的就仅仅卸载node去环境变量里面看一下有没有跟nvm和node相关的东西了,有的话全......
  • 【题解】CF1854A2 Dual (Hard Version)
    你考虑我们A1只需要通过自加凑一个最大的数,然后将所有的数都变成正数,最后做一次前缀和即可。(不懂可以看看落谷题解)好,我们现在去看HardVersion的\(31\)次操作怎么分配:前缀和(全为正)/后缀和(全为负)——\(19\)次还剩下\(12\)次,不知道该怎么做。我们的目标便变......