Bzoj 4371: [IOI2015]sorting排序 二分
似乎很久没写题解了...
这题是校里胡策的时候的题,比赛因为评测机有点慢+自己代码常数大没快读...被卡t了,但是bzoj上还是A了的...,因为bzoj时限比较宽可以不卡常。
题解:
首先可以发现答案与操作顺序是无关的,也就是说,可以钦定答案就是x次操作,然后让先手的x次先全换了,然后再考虑我要怎么换,才能在最少次数内换成升序。
于是就可以直接枚举答案x,然后判一下x是否可行。
考虑如何判断,问题会变成,给定一个a序列,每次可以交换两个数,问最少交换多少次可以换成升序,也就是变成a[i]=i。
这是一个经典问题,考虑 i 必须换到 a[i] 的位置,于是就直接一直跳就好了,最后会变成若干个环。
一个环中如果有n个数,那么必须需要也只需要 n-1 次就可以换成 a[i]=i 的情况。
因此最少的交换次数就是每个环大小-1加起来,可以在 O(n) 的时间复杂度下完成判断。
那么枚举答案+判断答案是 O(n^2) 的,没得聊。
其实答案是可以二分的,满足二分性质。
证明:即证明如果答案为x可行,那么答案为x+1也必然可行,如果x可行,那么我花x次操作变成升序,然后第x+1次操作,先手怎么换,我就再换回去,序列依旧是升序的。
那么效率就是 O(nlogn) 的
虽然答案与顺序无关,但是操作方案是和顺序有关的。
转换一下思路,交换两个数,可以理解成两个位置交换,但是其实也可以理解成两个数字交换。
而位置和顺序有关,因为先手换完之后我本来想换的位置就会变了。
但是数字和顺序是没关系的,所以我记录一下我环中交换的那些数。
然后按题意模拟,每次维护一下now[i]表示 i 这个数现在的位置,于是就变得很简单了...
注意一下两个人都操作一次才算完,不能先手操作完后是升序的我就不操作了。
所以如果可以不操作的,要拿 0 0补满
#include<cstdio>
#include<algorithm>
#define maxn 200050
using namespace std;
int n;
int v[maxn],now[maxn];
int a[maxn],b[maxn],x[maxn*],y[maxn*];
struct qnode{
int x,y;
}q[maxn*];
int check(int m){
for (int i=;i<n;i++)
b[i]=a[i],v[i]=;
for (int i=;i<=m;i++)
swap(b[x[i]],b[y[i]]);
int need=;
for (int i=;i<n;i++)
if (!v[i]){
int x=i;
while (!v[x]){
v[x]=;
if (!v[b[x]]) {
q[++need].x=b[x];
q[need].y=b[b[x]];
}
x=b[x];
}
}
return need;
}
int main(){
// freopen("game.in","r",stdin);
// freopen("game.out","w",stdout);
scanf("%d",&n);
for (int i=;i<n;i++)
scanf("%d",&a[i]),now[a[i]]=i;
int Q;
scanf("%d",&Q);
for (int i=;i<=Q;i++)
scanf("%d%d",&x[i],&y[i]);
int l=,r=Q;
while (l<=r){
int m=(l+r)>>;
if (check(m)>m) l=m+;else r=m-;
}
int need=check(l);
printf("%d\n",l);
for (int i=;i<=need;i++){
swap(a[x[i]],a[y[i]]);
now[a[x[i]]]=x[i];
now[a[y[i]]]=y[i];
printf("%d %d\n",now[q[i].x],now[q[i].y]);
swap(a[now[q[i].x]],a[now[q[i].y]]);
now[a[now[q[i].x]]]=now[q[i].x];
now[a[now[q[i].y]]]=now[q[i].y];
}
for (int i=need+;i<=l;i++)
printf("0 0\n");
return ;
}
4371
Bzoj 4371: [IOI2015]sorting排序 二分的更多相关文章
- bzoj 4552 [Tjoi2016&Heoi2016]排序 (二分答案 线段树)
题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=4552 题意: 给你一个1-n的全排列,m次操作,操作由两种:1.将[l,r]升序排序,2 ...
- BZOJ 4552 [Tjoi2016&Heoi2016]排序 | 二分答案 线段树
题目链接 题面 题目描述 在2016年,佳媛姐姐喜欢上了数字序列.因而他经常研究关于序列的一些奇奇怪怪的问题,现在他在研究一个难题,需要你来帮助他.这个难题是这样子的:给出一个1到n的全排列,现在对这 ...
- bzoj 4552 [Tjoi2016&Heoi2016]排序——二分答案
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4552 二分答案,把 >= mid 的设成1.< mid 的设成0,之后排序就变成 ...
- bzoj 4552: [Tjoi2016&Heoi2016]排序——二分+线段树
Description 在2016年,佳媛姐姐喜欢上了数字序列.因而他经常研究关于序列的一些奇奇怪怪的问题,现在他在研究一个难题 ,需要你来帮助他.这个难题是这样子的:给出一个1到n的全排列,现在对这 ...
- UVA.10474 Where is the Marble ( 排序 二分查找 )
UVA.10474 Where is the Marble ( 排序 二分查找 ) 题意分析 大水题一道.排序好找到第一个目标数字的位置,返回其下标即可.暴力可过,强行写了一发BS,发现错误百出.应了 ...
- bzoj 4552: [Tjoi2016&Heoi2016]排序【二分+线段树】
二分值mid,然后把>=mid的赋值为1,其他赋值为0,每次排序就是算出区间内01的个数,然后分别把0和1放到连续的一段内,这些都可以用线段树来维护 二分的判断条件是操作完之后q位置上是否为1 ...
- BZOJ 4552: [Tjoi2016&Heoi2016]排序 线段树 二分
目录 此代码是个假代码,只能糊弄luogu,以后再改,路过大佬也可以帮一下辣 update 10.6 此代码是个假代码,只能糊弄luogu,以后再改,路过大佬也可以帮一下辣 /* //fang zhi ...
- BZOJ.4552.[HEOI2016/TJOI2016]排序(线段树合并/二分 线段树)
题目链接 对于序列上每一段连续区间的数我们都可以动态开点建一棵值域线段树.初始时就是\(n\)棵. 对于每次操作,我们可以将\([l,r]\)的数分别从之前它所属的若干段区间中分离出来,合并. 对于升 ...
- BZOJ 4552 [Tjoi2016&Heoi2016]排序 ——线段树 二分答案
听说是BC原题. 好题,二分答案变成01序列,就可以方便的用线段树维护了. 然后就是区间查询和覆盖了. #include <map> #include <cmath> #inc ...
随机推荐
- Android 使用动画效果后的控件位置处理 类似系统通知栏下拉动画
Android的动画的使用,请参考.Android的动画,在设计方面,我有点不太理解,觉得这样搞很怪,因为在控件动画后,即使设置了停留在动画结束时的位置,我们也确实看到了控件停在那个位置,但其实该控件 ...
- 有关velocity的资料(等待整理)
proxy-target-class="true" 与proxy-target-class="false"的区别: proxy-target-class属性值决 ...
- 重新来认识你的老朋友Spring框架
欢迎查看Java开发之上帝之眼系列教程,如果您正在为Java后端庞大的体系所困扰,如果您正在为各种繁出不穷的技术和各种框架所迷茫,那么本系列文章将带您窥探Java庞大的体系.本系列教程希望您能站在上帝 ...
- Javascript一(变量,数据类型,正则表达式,数据,语句)
本文章适合具有一定程序编程语言基础的人士阅读,最好学完Java基础再来阅读本文章更容易理解语言初学者会看起来比较费劲,不易理解 一.导入脚本 在html导入Javascript的格式是: <sc ...
- 从零开始写JavaWeb框架(第四章节的AOP)
使用"链式代理"实现 AOP 本文是<轻量级 Java Web 框架架构设计>的系列博文. 大家是否还记得<Proxy 那点事儿>中提到的 CGLib ...
- git-【十】忽略文件
1.在Git工作区的根目录下创建一个特殊的.gitignore文件,然后把要忽略的文件名填进去,Git就会自动忽略这些文件. 不需要从头写.gitignore文件,GitHub已经为我们准备了各种配置 ...
- jvm启动
首先使用 Java 命令启动JVM 其次进行JVM配置的装载——根据当前路径和系统的版本去寻找jvm.cfg文件,装载配置. 每种需要java虚拟机的软件,都会带一个jvm.cfg.然后jvm.cfg ...
- html基础之css标签
css style: 里面的写的就叫做css,每一个样式的间隔用英文分号, 全部相同的时候引用class. css有三种写法: 1.在head标签中增加style标签,在style标签中去写css样式 ...
- 深入理解Nginx
nginx概述 nginx是一款自由的.开源的.高性能的HTTP服务器和反向代理服务器:同时也是一个IMAP.POP3.SMTP代理服务器:nginx可以作为一个HTTP服务器进行网站的发布处理,另外 ...
- Python常用模块(logging&re&时间&random&os&sys&shutil&序列化&configparser&&hashlib)
一. logging(日志模块) 二 .re模块 三. 时间模块 四. random模块 五. os模块 六. sys模块 七. shutil模块 八. 序列化模块(json&pickle&a ...