ARC165F题解
前言
简要题意
给你一个长度为
数据范围:
思路
做题的时候首先应该考虑题目性质,可以从手玩样例开始。因为最后并没有让你求最少操作次数,所以我们只用讨论数与数之间的关系。我们考虑最简单的情况:假设现在序列中只有
对于第一种和第六种情况我们可以不用考虑,因为需要保证操作数最小。然后这四种情况实际只有两种本质不同,我们将他们抓出来。假设有两数
对于第一个情况,我们只需将中间两数交换即可。而第二种,我们既可以将第一个数交换到第三个位置,也可以将最后一个数交换到第二个位置。也就是说:第一种情况下数的位置决定最后顺序;而第二种情况下数的大小决定了最后顺序。
现在考虑扩展这两种情况,对于数列中任意的两数
总结一下:
可是直接建图跑是
我们横着切一刀把平面分成两部分,在分割线上建一些虚点。对于下面的实点垂直向上连边,上面的实点从下面虚点往上连边,然后虚点之间从左往右连。若每次在中间切最多切出
其实我们只需要对实点用优先队列,对于虚点我们不关心他们的具体顺序,所以开一个普通队列存虚点,另一个优先队列存实点,每次先把所有普通队列的点拓扑完再去拓扑优先队列就行。时间复杂度是
代码
void cdq(int l, int r){
if(l == r)return; int mid = l + r >> 1, lim = a[mid].l;
cdq(l, mid), cdq(mid + 1, r);
int i = l, j = mid + 1, k = l;
while(i <= mid and j <= r)a[i].r < a[j].r ? b[k++] = a[i++] : b[k++] = a[j++];
while(i <= mid)b[k++] = a[i++]; while(j <= r)b[k++] = a[j++];
for(int i = l; i <= r; ++i){
a[i] = b[i], ++nd; if(i ^ l)e[nd - 1].pb(nd), ++in[nd];
if(a[i].l <= lim)e[a[i].id].pb(nd), ++in[nd];
else e[nd].pb(a[i].id), ++in[a[i].id];
}
}
void upd(int x){
if(in[x])return;
x <= n ? q.push(x) : qc.push(x);
}
signed main(){
freopen("swap.in", "r", stdin);
freopen("swap.out", "w", stdout);
n = rd(), nd = n << 1;
for(int i = 1; i <= nd; ++i){
int x = rd();
if(a[x].l)a[x].r = i; else a[x].l = i, a[x].id = x;
}
sort(a + 1, a + 1 + n); nd = n;
cdq(1, n);
for(int i = 1; i <= nd; ++i)upd(i);
while(! q.empty() or ! qc.empty()){
while(! qc.empty()){
int u = qc.front(); qc.pop();
for(int v : e[u])--in[v], upd(v);
}
if(q.empty())return 0;
int u = q.top(); q.pop();
pc(u), putchar(' '), pc(u), putchar(' ');
for(int v : e[u])--in[v], upd(v);
}
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· winform 绘制太阳,地球,月球 运作规律
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)