ARC165F题解
前言
\(2024.10.19\) 日校测 \(T4\),思维太庙,被薄纱了,遂哭弱,写题解以记之。
简要题意
给你一个长度为 \(2n\) 的序列满足 \(\forall a_i\in[1,n]\),其中 \(1\) 到 \(n\) 每个数都出现了两次,现在需要把相同的两个数排到一起,每次操作只能交换相邻两个数,在保证操作次数最小的情况下求出现在序列的最小字典序。
数据范围:\(1\le n\le2\times10^5\)。
思路
做题的时候首先应该考虑题目性质,可以从手玩样例开始。因为最后并没有让你求最少操作次数,所以我们只用讨论数与数之间的关系。我们考虑最简单的情况:假设现在序列中只有 \(1\) 和 \(2\) 各两个,一共存在六种可能的情况。我们先将他们列出来:
- \(1122\)
- \(1212\)
- \(1221\)
- \(2112\)
- \(2121\)
- \(2211\)
对于第一种和第六种情况我们可以不用考虑,因为需要保证操作数最小。然后这四种情况实际只有两种本质不同,我们将他们抓出来。假设有两数 \(A,B\) 位置关系如下:
- \(ABAB\)
- \(ABBA\)
对于第一个情况,我们只需将中间两数交换即可。而第二种,我们既可以将第一个数交换到第三个位置,也可以将最后一个数交换到第二个位置。也就是说:第一种情况下数的位置决定最后顺序;而第二种情况下数的大小决定了最后顺序。
现在考虑扩展这两种情况,对于数列中任意的两数 \(A,B\),如果满足 \(A\dots B\dots A\dots B\) 的形式,我一定会让 \(A\) 排在 \(B\) 前面;如果满足 \(A\dots B\dots B\dots A\) 的形式,我就会去考虑两个数之间的大小关系。
总结一下:
\(\forall x\in[1,n]\),设 \(a_x\) 表示其第一次出现的位置,\(b_x\) 表示第二次出现的位置,如满足偏序:\(a_i\le a_j,b_i\le b_j\) 则 \(i\) 在 \(j\) 之前。所以把这些偏序抽象成一张图跑拓扑排序,拓扑时让数字小的点尽量先跑就能满足第二种情况。
可是直接建图跑是 \(O(n^2)\) 的,考虑优化。我们把每一个关于 \(i\) 的二元组 \((a_i,b_i)\) 看成平面内的点,若 \(i,j\) 之间连边则需要满足上述偏序。我们可以考虑分治建图,也就是类似 \(\text{cdq}\) 的过程,具体见下图:

我们横着切一刀把平面分成两部分,在分割线上建一些虚点。对于下面的实点垂直向上连边,上面的实点从下面虚点往上连边,然后虚点之间从左往右连。若每次在中间切最多切出 \(\log n\) 层,所以只有 \(n\log n\) 个点和边。但是拓扑的时候如果用优先队列是两只 \(\log\) 的,考虑继续优化。
其实我们只需要对实点用优先队列,对于虚点我们不关心他们的具体顺序,所以开一个普通队列存虚点,另一个优先队列存实点,每次先把所有普通队列的点拓扑完再去拓扑优先队列就行。时间复杂度是 \(O(n\log n)\) 的,因为实点只有 \(n\) 个。
代码
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;
}
ARC165F题解的更多相关文章
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- noip2016十连测题解
以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ...
- 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解
题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...
- 2016ACM青岛区域赛题解
A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Jav ...
- poj1399 hoj1037 Direct Visibility 题解 (宽搜)
http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...
- 网络流n题 题解
学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...
- CF100965C题解..
求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...
- JSOI2016R3 瞎BB题解
题意请看absi大爷的blog http://absi2011.is-programmer.com/posts/200920.html http://absi2011.is-programmer.co ...
随机推荐
- 【前端】display:inline-block中间的间隙
父元素宽度800px 两个并列子元素宽度设为50%,并且使用了display:inline-block 结果一运行,两个元素不在同一行? 我查审元素,两个子元素div的宽度都是400,也没有边框,边距 ...
- 【XML】学习笔记第三章-namesapce
目录 命名空间 命名空间概述 命名空间语法 命名空间的声明 命名空间作用域 对命名空间的使用 元素对命名空间的使用 属性对命名空间的使用 DTD对命名空间的支持 命名空间 命名空间概述 标记中出现了同 ...
- 【爬虫】爬虫简单举例(三种写法) 涉及requests、urllib、bs4,re
目录 写法1:requests + re 写法2:urllib + re 方法3:request + bs4 补充.我在发了这篇文章之后,后面又要爬这个论坛.然后我发现我没有认真观察那个网页的源码,所 ...
- 利用shell中awk和xargs以及sed将多行多列文本中某一列合并成一行
一.问题描述最近需要利用Shell将多行多列文本中某一列,通过指定的分隔符合并成一行.假设需要处理的文本如下: 我们主要处理的是,将用户名提取处理,合并成一行,并通过逗号进行分隔.最终的格式如下: & ...
- Mybatis-plus关于代码生成器的使用
1.添加依赖 2.在test包下创建一个CodeGet类,实现生成代码的功能.注意:全局配置.数据源配置一定要和自己的电脑配置一致! 3.执行CodeGet类中的main方法.打印台有如下图提示字样, ...
- [转]OpenCV_Find Basis F-Matrix and computeCorrespondEpilines(获取一对图像的基础矩阵及对应极线)
代码如下: // BasisMatrixCalculate.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include <ios ...
- [转]来,让我们一起来盘盘 Nodejs 环境变量(process.env)
首先 process.env 是什么? node环境变量: process 是node的全局变量,类似浏览器的window: env 是process的一个属性. 官方解释:process 对象是一个 ...
- 基于开源IM即时通讯框架MobileIMSDK:RainbowChat v11.7版已发布
关于RainbowChat RainbowChat是一套基于开源IM聊天框架 MobileIMSDK 的产品级移动端IM系统.RainbowChat源于真实运营的产品,解决了大量的屏幕适配.细节优化. ...
- MongoDB:文章评论系统模拟
- vscode python remote debug极速入门
本文适用范围 主要适用于debug python 程序,尤其是深度学习刚入门需要使用remote 连接到linux进行程序运行,想调试一下的同学. 当然非深度学习也可以参考食用本文哈哈哈. 极速入门版 ...