Luogu1155 NOIP2008 双栈排序


题目描述

Tom最近在研究一个有趣的排序问题。如图所示,通过 2个栈 S1 和 S2 ,Tom希望借助以下 44 种操作实现将输入序列升序排序。



操作 a 如果输入序列不为空,将第一个元素压入栈 S1

操作 b 如果栈 S1 不为空,将 S1 栈顶元素弹出至输出序列

操作 c 如果输入序列不为空,将第一个元素压入栈 S2

操作 d 如果栈 S2 不为空,将 S2 栈顶元素弹出至输出序列

如果一个 1-n的排列P可以通过一系列操作使得输出序列为 1,2,…,(n-1),n,Tom就称 PP 是一个“可双栈排序排列”。例如 (1,3,2,4)就是一个“可双栈排序序列”,而 (2,3,4,1)不是。下图描述了一个将 (1,3,2,4)排序的操作序列:

&lt;a,c,c,b,a,d,d,b&gt;" role="presentation"><a,c,c,b,a,d,d,b><a,c,c,b,a,d,d,b>



当然,这样的操作序列有可能有几个,对于上例 (1,3,2,4),

输入输出格式

输入格式:

第一行是一个整数 n 。

第二行有 n个用空格隔开的正整数,构成一个 1-n的排列

输出格式:

共一行,如果输入的排列不是“可双栈排序排列”,输出数字 0;否则输出字典序最小的操作序列,每两个操作之间用空格隔开,行尾没有空格。

输入输出样例

输入样例#1:

4
1 3 2 4

输出样例#1:

a b a a b b a b

输入样例#2:

4
2 3 4 1

输出样例#2:

0

输入样例#3:

3
2 3 1

输出样例#3:

a c a b b d

说明

30% 的数据满足: n≤10

50% 的数据满足: n≤50

100% 的数据满足: n≤1000


一开始没看出来是二分图染色,网上好像有方法模拟+贪心过的,ORZ

我们可以发现如果对于k

//yangkai
#include<bits/stdc++.h>
using namespace std;
#define N 1010
int n,a[N],mink[N],col[N],g[N][N];
bool dfs(int x){
for(int i=1;i<=n;i++)
if(g[i][x]){
if(col[i]!=-1){
if(col[x]!=col[i])continue;
return 0;
}
col[i]=1-col[x];
dfs(i);
}
return 1;
}
int main(){
memset(mink,0x3f,sizeof(mink));
memset(col,-1,sizeof(col));
scanf("%d",&n);
for(int i=1;i<=n;i++)scanf("%d",&a[i]);
for(int i=n;i>=1;i--)mink[i]=min(mink[i+1],a[i]);
for(int i=1;i<=n;i++)
for(int j=i+1;j<=n;j++)
if(a[i]<a[j]&&mink[j+1]<a[i])
g[i][j]=g[j][i]=1;
for(int i=1;i<=n;i++)if(col[i]==-1){
col[i]=1;
if(!dfs(i)){
printf("0");
return 0;
}
}
int st=col[1],las=0;
stack<int> p1,p2;
for(int i=1;i<=n;i++){
if(col[i]==st){
while(!p1.empty()&&p1.top()<a[i]){
if(!p2.empty()&&p2.top()<p1.top()){las=p2.top();p2.pop();printf("d ");}
else{las=p1.top();p1.pop();printf("b ");}
}
p1.push(a[i]);
printf("a ");
}else{
while(!p2.empty()&&p2.top()<a[i]){
if(!p1.empty()&&p2.top()>p1.top()){las=p1.top();p1.pop();printf("b ");}
else{las=p2.top();p2.pop();printf("c ");}
}
while(!p1.empty()&&p1.top()==las+1){
las++;
p1.pop();
printf("b ");
}
p2.push(a[i]);
printf("c ");
}
}
while(!p1.empty()&&!p2.empty()){
if(p1.top()<=p2.top()){p1.pop();printf("b ");}
else{p2.pop();printf("d ");}
}
while(!p1.empty()){p1.pop();printf("b ");}
while(!p2.empty()){p2.pop();printf("d ");}
return 0;
}

Luogu1155 NOIP2008 双栈排序 【二分图染色】【模拟】的更多相关文章

  1. [luogu1155 NOIP2008] 双栈排序 (二分图染色)

    传送门 Description Input 第一行是一个整数 n . 第二行有 n 个用空格隔开的正整数,构成一个 1−n 的排列. Output 共一行,如果输入的排列不是"可双栈排序排列 ...

  2. NOIP2008双栈排序[二分图染色|栈|DP]

    题目描述 Tom最近在研究一个有趣的排序问题.如图所示,通过2个栈S1和S2,Tom希望借助以下4种操作实现将输入序列升序排序. 操作a 如果输入序列不为空,将第一个元素压入栈S1 操作b 如果栈S1 ...

  3. [NOIp2008] 双栈排序 (二分图染色 + 贪心)

    题意 给你一个长为 \(n\) 的序列 \(p\) ,问是否能够通过对于两个栈进行 push, pop(print) 操作使得最后输出序列单调递增(即为 \(1 \cdots n\) ),如果无解输出 ...

  4. LOJ P1155 双栈排序 二分图染色 图论

    https://www.luogu.org/problem/show?pid=P1155 题解: https://www.byvoid.com/zhs/blog/noip2008-twostack 开 ...

  5. Luogu1155 NOIP2008双栈排序(并查集)

    两个位置i和j上的元素不能被放进同一个栈的充要条件显然是存在k使i<j<k且ak<ai<aj.由此在保证合法的情况下贪心地放就是正确的了. 至于如何判断,可以记一下后缀最小值, ...

  6. [NOIP2008]双栈排序 【二分图 + 模拟】

    题目描述 Tom最近在研究一个有趣的排序问题.如图所示,通过2个栈S1和S2,Tom希望借助以下4种操作实现将输入序列升序排序. 操作a 如果输入序列不为空,将第一个元素压入栈S1 操作b 如果栈S1 ...

  7. $[NOIp2008]$双栈排序 栈/二分图/贪心

    \(Sol\) 先考虑单栈排序,怎么样的序列可以单栈排序呢?设\(a_i\)表示位置\(i\)是哪个数.\(\exist i<j<k\),都没有\(a_k<a_i<a_j\), ...

  8. Noip2008双栈排序

    [问题描述] 用两个栈使一个1...n的排列变得有序.一共有四个操作: A.stack1.push() 读入一个放入栈一 B.stack1.pop() 弹出栈一放入输出序列 C.stack2.push ...

  9. [题解] [NOIP2008] 双栈排序——关系的冲突至图论解法

    Problem 题目描述 Tom最近在研究一个有趣的排序问题.如图所示,通过2个栈S1和S2,Tom希望借助以下4种操作实现将输入序列升序排序. 操作a 如果输入序列不为空,将第一个元素压入栈S1 操 ...

随机推荐

  1. 百度编辑器(ueditor)@功能之获取坐标

    //获取百度编辑器的工具类 var domUtils = UE.dom.domUtils; //获取编辑器的坐标 var $ueditor_offset = $("#ueditor_0&qu ...

  2. Linux CentOS 7 安装字体库 & 中文字体

    前言 报表中发现有中文乱码和中文字体不整齐(重叠)的情况,首先考虑的就是操作系统是否有中文字体,在CentOS 7中发现输入命令查看字体列表是提示命令无效: 如上图可以看出,不仅没有中文字体,连字体库 ...

  3. HTML5 history API,创造更好的浏览体验

    HTML5 history API有什么用呢? 从Ajax翻页的问题说起 请想象你正在看一个视频下面的评论,在翻到十几页的时候,你发现一个写得稍长,但非常有趣的评论.正当你想要停下滚轮细看的时候,手残 ...

  4. Ghost:一款简约风格博客系统

    前言 本文将介绍一种最快速的创建Ghost博客系统的方法,并实现绑定二级域名到该博客系统.本文以本博客的“微博客”为例. 一键创建Ghost博客系统 Kite 是 Ghost 博客托管商,网址为:ht ...

  5. html合并单元格

    在合并的首位置加上colspan或者rowspan属性即可 code: <html>   <body>   <h4>横跨两列的单元格:</h4> < ...

  6. CodeForces - 91B单调队列

    有一个数列,对于每一个数,求比它小的在他右边距离他最远的那个数和他的距离 用单调队列做,维护单调队列时可采用如下方法,对于每一个数,如果队列中没有数,则加入队列,如果队列头的数比当前数大,则舍弃该数 ...

  7. Python之NumPy(axis=0 与axis=1)区分

    转自:http://blog.csdn.net/wangying19911991/article/details/73928172 https://www.zhihu.com/question/589 ...

  8. CodeForces 297A Parity Game (脑补题)

    题意 一个01串,可以有两种操作:①在末尾添加parity(a):②删除开头的一个字符.其中parity(a),当串中1的个数为奇数时为1,偶数时为0.问某个01串是否可以通过若干操作变成另一个01串 ...

  9. 华硕笔记本开机直接进入bios解决方法

    1.开机按esc键或F12,进入bios,打开BOOT标签,将Launch CSM改为Enabled,但此时的Launch CSM是Disabled,并且是灰色不可编辑状态. 2.按方向键切换到Sec ...

  10. IOS UI-模态视图

    一.简单介绍 除了push之外,还有另外一种控制器的切换方式,那就是Modal 任何控制器都能通过Modal的形式展⽰出来 Modal的默认效果:新控制器从屏幕的最底部往上钻,直到盖住之前的控制器为⽌ ...