The 2024 ICPC Kunming Invitational Contest K. Permutation(交互 期望)

在知乎内查看

题目

思路来源

题解

首先特判n=1的情况,其实也不用问

分治,假设当前解决到[l,r],要递归的vector是x,

维护两个vector L、R,代表下一步要在[l,mid]和[mid+1,r]分治的vector

每次将x random_shuffle后,取出vector尾部的两个u、v,

计分界点为mid,<=mid的全填u,>mid的全填v,看询问出的答案:

  1. 如果为0,说明都询问错了,则交换u、v所在位置,放进对应vector
  2. 如果为2,说明都询问对了,直接放入对应vector
  3. 否则为1,说明u和v位于一边,此时将v塞进del这个vector里,将u和v在并查集上合并,并把u塞回x

重复这个过程,直至x为空或只剩一个元素,

只剩一个元素时,L或R一定已经有元素,

和已经被询问出来的元素再一起询问一次,就能确定出这个元素该放进L还是R

代码中用to数组记录了是放进左边还是放进右边,

这样del里的元素,在并查集上找到其祖先时,可以用to数组确定其应该被放进L还是R

期望次数是6000的,实际跑得飞快,也没被卡掉

代码

cpp 复制代码
#include<bits/stdc++.h>
#include<iostream>
#include<cstdio>
#include<vector>
#include<queue>
#include<map>
#include<set>
using namespace std;
#define rep(i,a,b) for(int i=(a);i<=(b);++i)
#define per(i,a,b) for(int i=(a);i>=(b);--i)
typedef long long ll;
typedef double db;
typedef pair<ll,ll> P;
#define fi first
#define se second
#define pb push_back
#define dbg(x) cerr<<(#x)<<":"<<x<<" ";
#define dbg2(x) cerr<<(#x)<<":"<<x<<endl;
#define SZ(a) (int)(a.size())
#define sci(a) scanf("%d",&(a))
#define pt(a) printf("%d",a);
#define pte(a) printf("%d\n",a)
#define ptlle(a) printf("%lld\n",a)
#define debug(...) fprintf(stderr, __VA_ARGS__)
const int N=1e3+10;
int n,ans[N],q[N],par[N],to[N];
int find(int x){
    return par[x]==x?x:par[x]=find(par[x]);
}
int ask(){
    printf("0");
    rep(i,1,n){
        printf(" %d",q[i]);
    }
    printf("\n");
    fflush(stdout);
    int v;
    sci(v);
    return v;
}
void out(){
    printf("1");
    rep(i,1,n){
        printf(" %d",ans[i]);
    }
    printf("\n");
    fflush(stdout);
}
void sol(int l,int r,vector<int>x){
    //printf("l:%d r:%d ",l,r);
    //for(auto &v:x)printf("%d ",v);puts("");
    if(l==r){
        ans[l]=x[0];
        return;
    }
    for(auto &v:x)par[v]=v;
    int mid=(l+r)/2;
    vector<int>L,R,del;
    while(SZ(x)>1){
        random_shuffle(x.begin(),x.end());
        int u=x.back();x.pop_back();
        int v=x.back();x.pop_back();
        rep(i,1,n){
            if(i<=mid)q[i]=u;
            else q[i]=v;
        }
        int w=ask();
        if(!w)L.pb(v),R.pb(u),to[v]=0,to[u]=1;
        else if(w==2)L.pb(u),R.pb(v),to[u]=0,to[v]=1;
        else del.pb(v),x.pb(u),par[v]=u;
    }
    //printf("x:%d L:%d R:%d\n",SZ(x),SZ(L),SZ(R));
    if(SZ(x)==1){
        int u=x[0];
        if(SZ(L)){
            rep(i,1,n){
                if(i<=mid)q[i]=u;
                else q[i]=L[0];
            }
            int w=ask();
            if(!w)R.pb(u),to[u]=1;
            else L.pb(u),to[u]=0;
        }
        else if(SZ(R)){
            rep(i,1,n){
                if(i<=mid)q[i]=R[0];
                else q[i]=u;
            }
            int w=ask();
            if(!w)L.pb(u),to[u]=0;
            else R.pb(u),to[u]=1;
        }
        else{
            assert(false);
        }
    }
    for(auto &v:del){
        int fa=find(v);
        if(!to[fa])L.pb(v);
        else R.pb(v);
    }
    if(SZ(L))sol(l,mid,L);
    if(SZ(R))sol(mid+1,r,R);
}
void sol(){
    if(n==1){
        ans[1]=1;
        out();
        return;
    }
    vector<int>now;
    rep(i,1,n)now.pb(i);
    sol(1,n,now);
    out();
}
int main(){
    srand(time(NULL));
    sci(n);
    sol();
    return 0;
}
//2 3 4 1 5
相关推荐
方见华Richard14 小时前
递归对抗引擎RAE V4.0(AGI自主进化版)
经验分享·笔记·其他·交互·学习方法
devmoon17 小时前
如何使用 Web3.py 与 Polkadot Hub 进行交互
web3·区块链·智能合约·交互·web3.py·solidity·polkadot
方见华Richard17 小时前
递归对抗引擎(RAE)核心极简实现框架
人工智能·交互·学习方法·原型模式·空间计算
方见华Richard17 小时前
递归对抗引擎RAE V2.0(多智能体分布式对抗版)
人工智能·经验分享·交互·学习方法·原型模式
晚霞的不甘19 小时前
Flutter for OpenHarmony《淘淘购物》主页点击跳转功能详解:从 UI 响应到页面导航的完整实现
前端·flutter·ui·搜索引擎·前端框架·交互
●VON21 小时前
Flutter for OpenHarmony:基于软删除状态机与双轨数据管理的 TodoList 回收站安全体系实现
安全·flutter·交互·openharmony·跨平台开发·von
沛沛老爹1 天前
Web开发者实战:多模态Agent技能开发——语音交互与合成技能集成指南
java·开发语言·前端·人工智能·交互·skills
晚霞的不甘1 天前
Flutter for OpenHarmony《淘淘购物》 分类页:从静态 UI 到动态交互的全面升级
flutter·ui·前端框架·交互·鸿蒙
●VON2 天前
Flutter for OpenHarmony:基于可选描述字段与上下文感知渲染的 TodoList 任务详情子系统实现
学习·flutter·架构·交互·von
●VON2 天前
Flutter for OpenHarmony:基于可空截止日期与时间语义可视化的 TodoList 时间管理子系统实现
安全·flutter·交互·openharmony·跨平台开发