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 ...
随机推荐
- PHP获取POST的原始数据的方法
一般我们都用$_POST或$_REQUEST两个预定义变量来接收POST提交的数据.但如果提交的数据没有变量名,而是直接的字符串,则需要使用其他的方式来接收. 方法一: 使用全局变量$GLOBALS[ ...
- Rsync匿名访问漏洞
前言 前两天总结了互联网或者说IT公司内网常见的漏洞,然后决定针对还没学习过不了解的漏洞进行学习了解,所以准备一一针对来研习,今天是第一篇,立一个Flag,争取今年搞定,为啥说的这么艰难,因为平时工作 ...
- 深入理解javascript原型和闭包 摘要
一切(引用类型)都是对象,对象是属性的集合 对象都是通过函数创建的 隐式原型 Instanceof的判断队则是:沿着A的__proto__这条线来找,同时沿着B的prototype这条线来找,如果两条 ...
- SeaJS简介一:由来,特点以及优势
由来: 在软件开发过程中,模块化编程思想已经习以为常了,模块化编程不仅仅给开发团队带来效率方面上的好处,还能够让开发的项目或者产品维护成本大大降低. 那么,在WEB开发过程中JS脚本语言已经不可或缺了 ...
- oracle的分页查询,mabatis的sql配置
<select id="getCardcaseByPage" resultType="Cardcase" > select * from ( sel ...
- mybatis的<choose>和<when>、<otherwise>标签
SELECT<choose> <when test='timeType=="yy"'> TO_CHAR(REPORT_TIME,'yyyy') </w ...
- Hive格式化输出数据库和表详细信息
hive> desc database extended wx_test; OK wx_test hdfs://ns1/user/hive/warehouse/wx_test.db hadoop ...
- Python开发【模块】:re正则
re模块 序言: re模块用于对python的正则表达式的操作 '.' 默认匹配除\n之外的任意一个字符,若指定flag DOTALL,则匹配任意字符,包括换行 '^' 匹配字符开头,若指定flags ...
- 符合语言习惯的 Python 优雅编程技巧
Python最大的优点之一就是语法简洁,好的代码就像伪代码一样,干净.整洁.一目了然.要写出 Pythonic(优雅的.地道的.整洁的)代码,需要多看多学大牛们写的代码,github 上有很多非常优秀 ...
- in 和 or 的效率问题
select * from table where col in (2,3,4,5,6) select * from table where col=2 or col=3 or col=4 or co ...