题目

似乎很久没写题解了...

这题是校里胡策的时候的题,比赛因为评测机有点慢+自己代码常数大没快读...被卡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排序 二分的更多相关文章

  1. bzoj 4552 [Tjoi2016&Heoi2016]排序 (二分答案 线段树)

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=4552 题意: 给你一个1-n的全排列,m次操作,操作由两种:1.将[l,r]升序排序,2 ...

  2. BZOJ 4552 [Tjoi2016&Heoi2016]排序 | 二分答案 线段树

    题目链接 题面 题目描述 在2016年,佳媛姐姐喜欢上了数字序列.因而他经常研究关于序列的一些奇奇怪怪的问题,现在他在研究一个难题,需要你来帮助他.这个难题是这样子的:给出一个1到n的全排列,现在对这 ...

  3. bzoj 4552 [Tjoi2016&Heoi2016]排序——二分答案

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4552 二分答案,把 >= mid 的设成1.< mid 的设成0,之后排序就变成 ...

  4. bzoj 4552: [Tjoi2016&Heoi2016]排序——二分+线段树

    Description 在2016年,佳媛姐姐喜欢上了数字序列.因而他经常研究关于序列的一些奇奇怪怪的问题,现在他在研究一个难题 ,需要你来帮助他.这个难题是这样子的:给出一个1到n的全排列,现在对这 ...

  5. UVA.10474 Where is the Marble ( 排序 二分查找 )

    UVA.10474 Where is the Marble ( 排序 二分查找 ) 题意分析 大水题一道.排序好找到第一个目标数字的位置,返回其下标即可.暴力可过,强行写了一发BS,发现错误百出.应了 ...

  6. bzoj 4552: [Tjoi2016&Heoi2016]排序【二分+线段树】

    二分值mid,然后把>=mid的赋值为1,其他赋值为0,每次排序就是算出区间内01的个数,然后分别把0和1放到连续的一段内,这些都可以用线段树来维护 二分的判断条件是操作完之后q位置上是否为1 ...

  7. BZOJ 4552: [Tjoi2016&Heoi2016]排序 线段树 二分

    目录 此代码是个假代码,只能糊弄luogu,以后再改,路过大佬也可以帮一下辣 update 10.6 此代码是个假代码,只能糊弄luogu,以后再改,路过大佬也可以帮一下辣 /* //fang zhi ...

  8. BZOJ.4552.[HEOI2016/TJOI2016]排序(线段树合并/二分 线段树)

    题目链接 对于序列上每一段连续区间的数我们都可以动态开点建一棵值域线段树.初始时就是\(n\)棵. 对于每次操作,我们可以将\([l,r]\)的数分别从之前它所属的若干段区间中分离出来,合并. 对于升 ...

  9. BZOJ 4552 [Tjoi2016&Heoi2016]排序 ——线段树 二分答案

    听说是BC原题. 好题,二分答案变成01序列,就可以方便的用线段树维护了. 然后就是区间查询和覆盖了. #include <map> #include <cmath> #inc ...

随机推荐

  1. VScode之JavaScript Snippet Pack

    一个片段包 使用例如: cl 回车或者tab键,就可以完整的打出console.log("") 还有很多快捷功能: 参考: https://marketplace.visualst ...

  2. Python GUI--Tkinter实践

    之前写了Testlink自动执行程序,现使用Tkinter加上GUI试试,想要实现如下图功能 可以实现通过选择要执行的url及报告url自动执行用例,或可以直接写报告结果内容 因项目原因,只列出部分代 ...

  3. Redis对于key的操作命令

     del key1 key2 ... Keyn 作用: 删除1个或多个键 返回值: 不存在的key忽略掉,返回真正删除的key的数量 rename key newkey 作用: 给key赋一个新的ke ...

  4. github团队协作教程

    跟着笔者魔鬼般的步伐,我们一起来瞅瞅一个团队协作的任务如何进行版本管理吧~ 要跟上哦~ =============================================== 首先我们先来看下 ...

  5. python2在安装pywin32后出现ImportError: DLL load failed 解决方法

    python2在安装pywin32后出现ImportError: DLL load failed 解决方法 在python2中有时候会出现: import win32api   ImportError ...

  6. Oracle性能优化之Oracle里的统计信息

    一.什么是统计信息 oracle数据库里的统计信息是如下的一组数据:他们存储在数据字典里,且从多个维度描述了oracle数据库数据对象的详细信息. oracle数据库里的统计信息主要分为以下6种情况: ...

  7. avaScript 的基础学习(一)

    JavaScript概述 JavaScript由三部分组成: JavaScript的基础 JS的引入方式: JS的数据类型: 运算符: 流程控制: JavaScript的对象 BOM对象 DOM Ev ...

  8. vue.js(四)

    由于组件内容太多又特别关键,我决定在官网教程的基础上,加上自己的理解,针对每个内容详细记录一下  1.注册组件 ①全局注册 //首先创建组件 Vue.component('blog-post', { ...

  9. Mysql覆盖索引与延迟关联

    延迟关联:通过使用覆盖索引查询返回需要的主键,再根据主键关联原表获得需要的数据.   为什innodb的索引叶子节点存的是主键,而不是像myisam一样存数据的物理地址指针? 如果存的是物理地址指针不 ...

  10. 前端 html head meta

    META(Metadata information) 提供有页面的元信息 例如:页面编码.刷新.跳转.针对搜索引擎和更新频道的描述和关键词 1.另外一种编码写法 <meta http-equiv ...