一道挺有意思的题目,在这里记录一下。

题目大意

给你一个长度为 \(n\) 的排列,每一次你可以取出相邻的两个数将其放在答案序列的开头,最后问你字典序最小的答案序列是什么。

题解

由于最后是求字典序最小,所以我们肯定需要倒序求解,所以我们需要考虑如何取才能使得剩下的序列合法。

如果我们在区间 \([L,R]\) 当前选择点 \(l\) 和点 \(r\) ,那么最后的区间很显然会被我们分成三部分, \([L,l-1]\) , \([l+1,r-1]\) 和 \([r+1,R]\) 。而且这三个区间的长度必须为偶数才可以满足区间同时也可分。我们就可以用一个优先队列来模拟一下,每一次取出当前所以区间中可以选择字典最小的点对的区间,再将其拆分为三个区间。不断重复此操作直到最后所有的数都被取出即可。

代码

#include<bits/stdc++.h>
using namespace std;
const int N=2e5+5;
int n,a[N],b[N];
struct Point{int data,loc;};
bool operator < (const Point a,const Point b){return a.data<b.data;}
struct Seg_Tree
{
struct Node{Point data;}tr[N<<2];
void up(int u){tr[u].data=min(tr[u<<1].data,tr[u<<1|1].data);}
void build(int u,int l,int r,int a[])
{
if(l==r)
{
tr[u].data.data=a[l];
tr[u].data.loc=l;
return ;
}
int mid=(l+r)>>1;
build(u<<1,l,mid,a);
build(u<<1|1,mid+1,r,a);
up(u);
}
void chg(int u,int l,int r,int x)
{
if(l==r){tr[u].data.data=1e9+7;return ;}
int mid=(l+r)>>1;
if(x<=mid) chg(u<<1,l,mid,x);
else chg(u<<1|1,mid+1,r,x);
up(u);
}
Point query(int u,int l,int r,int x,int y)
{
if(x<=l&&r<=y) return tr[u].data;
int mid=(l+r)>>1;Point res={1000000007,0};
if(x<=mid) res=min(res,query(u<<1,l,mid,x,y));
if(y>mid) res=min(res,query(u<<1|1,mid+1,r,x,y));
return res;
}
}t[2];//0为偶,1为奇
struct Data{int l,r;Point data;};
bool operator < (const Data a,const Data b){return a.data.data>b.data.data;}
priority_queue<Data> q;
vector<int> res;
int main()
{
cin>>n;
for(int i=1,x;i<=n;++i)
{
scanf("%d",&x);
if(i&1) a[i]=1e9+7,b[i]=x;
else a[i]=x,b[i]=1e9+7;
}
t[0].build(1,1,n,a);
t[1].build(1,1,n,b);
Data tmp;
tmp.l=1,tmp.r=n;
tmp.data=t[1].query(1,1,n,1,n);
q.push(tmp);
for(int i=1;i<=n/2;++i)
{
tmp=q.top(),q.pop();
int tag=tmp.l%2;
// printf("%d\n",tag);
Point L=tmp.data,R=t[tag^1].query(1,1,n,L.loc+1,tmp.r);
// printf("%d %d %d %d\n",tmp.l,tmp.r,L.data,R.data);
res.push_back(L.data),res.push_back(R.data);
Data now;
now.l=tmp.l,now.r=L.loc-1;
if(now.l<=now.r)
{
now.data=t[tag].query(1,1,n,now.l,now.r);
q.push(now);
}
now.l=L.loc+1,now.r=R.loc-1;
if(now.l<=now.r)
{
now.data=t[tag^1].query(1,1,n,now.l,now.r);
q.push(now);
}
now.l=R.loc+1,now.r=tmp.r;
if(now.l<=now.r)
{
now.data=t[tag].query(1,1,n,now.l,now.r);
q.push(now);
}
}
for(int i=0;i<(int)res.size();++i) printf("%d ",res[i]);
printf("\n");
return 0;
}

AT2688 [ARC080C] Young Maids的更多相关文章

  1. AtCoder Regular Contest 080 (ARC080) E - Young Maids 线段树 堆

    原文链接http://www.cnblogs.com/zhouzhendong/p/8934377.html 题目传送门 - ARC080 E - Young Maids 题意 给定一个长度为$n$的 ...

  2. 【AtCoder Regular Contest 080E】Young Maids [堆][线段树]

    Young Maids Time Limit: 50 Sec  Memory Limit: 512 MB Description 给定一个排列,每次选出相邻的两个放在队头,要求字典序最小. Input ...

  3. AtCoder Regular Contest 080 E - Young Maids

    地址:http://arc080.contest.atcoder.jp/tasks/arc080_c 题目: E - Young Maids Time limit : 2sec / Memory li ...

  4. Young Maids

    E - Young Maids Time limit : 2sec / Memory limit : 256MB Score : 800 points Problem Statement Let N  ...

  5. Atcoder arc080E Young Maids(线段树+优先队列)

    给出一个n排列,每次可以选择相邻的两个数字放在新的排列首部,问最后形成的新的排列字典序最小是? 考虑新排列的第一个数字,则应是下标为奇数的最小数,下标不妨设为i.第二个数字应该下标大于i且为偶数的最小 ...

  6. 【递归】【线段树】【堆】AtCoder Regular Contest 080 E - Young Maids

    给你一个1~n的排列p,n是偶数,每次从中任选一对相邻的数出来,插到排列q的开头,如此循环,问你所能得到的字典序最小的排列q. 我们先确定q开头的两个数q1,q2,q1一定是p的奇数位的最小的数,而q ...

  7. 【Atcoder】ARC 080 E - Young Maids

    [算法]数学+堆 [题意]给定n个数的排列,每次操作可以取两个数按序排在新序列的头部,求最小字典序. [题解] 转化为每次找字典序最小的两个数按序排在尾部,则p1和p2的每次选择都必须满足:p1在当前 ...

  8. AtCoder Regular Contest 080 E:Young Maids

    题目传送门:https://arc080.contest.atcoder.jp/tasks/arc080_c 题目翻译 给你一个\(n\)的排列\(p\),一个空序列\(q\),你每次可以从\(p\) ...

  9. Atcoder 乱做

    最近感觉自己思维僵化,啥都不会做了-- ARC103 F Distance Sums 题意 给定第 \(i\) 个点到所有点的距离和 \(D_i\) ,要求构造一棵合法的树.满足第 \(i\) 个点到 ...

随机推荐

  1. shell 脚本之set 命令(转)

    服务器的开发和管理离不开 Bash 脚本,掌握它需要学习大量的细节. set命令是 Bash 脚本的重要环节,却常常被忽视,导致脚本的安全性和可维护性出问题.本文介绍它的基本用法,让你可以更安心地使用 ...

  2. 判断机器是big-endian、little-endian

    联合体union和大小端(big-endian.little-endian):下边示范了一种用途,代表四个含义的四个变量,但是可以用一个int来操作,直接int赋值,无论内存访问(指针大小的整数倍,访 ...

  3. synchronized 到底该不该用?

    我是风筝,公众号「古时的风筝」,一个兼具深度与广度的程序员鼓励师,一个本打算写诗却写起了代码的田园码农! 文章会收录在 JavaNewBee 中,更有 Java 后端知识图谱,从小白到大牛要走的路都在 ...

  4. CSS浮动和清除浮动

    1.浮动float div配合float浮动来做页面的布局,浮动最常用的地方就是用来做布局.只有一个标签设置了浮动float属性,就会变成块级标签. <!DOCTYPE html> < ...

  5. RTP协议解析及H264/H265 音视频RTP打包分析

    一 概述 实时传输协议(Real-time Transport Protocol或简写RTP)是一个网络传输协议,它是由IETF的多媒体传输工作小组1996年在RFC 1889中公布的. RTP协议详 ...

  6. WIN10—更改电脑桌面路径

    电脑默认的桌面路径一般都在C盘,而我们又特别喜欢把文件都放在桌面,因为桌面既方便又好找.可时间久了,桌面文件会越来越多,C盘空间会越来越小,会拖慢系统速度.怎么把系统桌面路径设置在非C盘呢?本期教程将 ...

  7. 精尽 MyBatis 源码分析 - MyBatis 初始化(三)之 SQL 初始化(上)

    该系列文档是本人在学习 Mybatis 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释(Mybatis源码分析 GitHub 地址.Mybatis-Spring 源码分析 GitHub ...

  8. Vegas媒体生成器是什么,有什么作用

    在专业视频剪辑软件-Vegas的界面中,有一个媒体生成器的界面,此界面包含HitFilm Light Flares,Pro Type Titler,测试图案,纯色,棋盘格,色彩渐变,噪声纹理,致谢字幕 ...

  9. 从数据仓库双集群系统模式探讨,看GaussDB(DWS)的容灾设计

    摘要:本文主要是探讨OLAP关系型数据库框架的数据仓库平台如何设计双集群系统,即增强系统高可用的保障水准,然后讨论一下GaussDB(DWS)的容灾应该如何设计. 当前社会.企业运行当中,大数据分析. ...

  10. 2016年第七届蓝桥杯【C++省赛B组】F、G、H、J 题解

    F. 方格填数 #深搜 题意 有\(10\)个格子,填入0~9的数字.要求:连续的两个数字不能相邻.(左右.上下.对角都算相邻),求可能的填数方案数. +--+--+--+ | | | | +--+- ...