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. linux一键安装nginx脚本

    #!/bin/sh echo "----------------------------------start install nginx ------------------------- ...

  2. Extjs 分页多选的实现

    Extjs 版本 6.X 单页面的多选,没有任何问题. 直接使用 Grid的配置项进行绑定即可获取: xtype: 'grid', bind: { selection: '{checkedRecord ...

  3. 缓存技术内部交流_05_Cache Through

    参考资料: http://www.ehcache.org/documentation/3.2/caching-patterns.html http://www.ehcache.org/document ...

  4. ES5下的React

    按照官方推荐的思路,React使用标准的ES6标准的语法.比如说创建一个类: class Greeting extends React.Component { render() { return &l ...

  5. 理解JAVA MQ消息中间件

    MQ的几种消息传递方式 发布订阅模式 发布订阅模式有点类似于我们日常生活中订阅报纸.每年到年尾的时候,邮局就会发一本报纸集合让我们来选择订阅哪一个.在这个表里头列了所有出版发行的报纸,那么对于我们每一 ...

  6. play的过滤类怎么实现继承问题

    原文: Example: public class Secure extends Controller {          @Before     static void checkAuthenti ...

  7. ContentPresenter元素

    一个内容控件 分解它的“结构树”,肯定能够看到ContentPresenter“元素”,该元素的功能:用来为“内容控件”显示“Content”

  8. CodeForces - 91B单调队列

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

  9. js Math对象的常用方法

    1,基本方法: Math.round();向上四舍五入. Math.ceil();向上取整,有小数就整数部分加1 Math.floor(5/2) ;向下取整 Math.abs();返回绝对值: Mat ...

  10. Jdev 本地RUN页面时候,将异常直接显示出来,而不是乱码

    本地运行页面时,经常会遇到以下错误 oracle.jbo.JboException: JBO-29000: JBO-29000: JBO-26028: ???? MemberAttributesDis ...