P1155 双栈排序
题目描述
Tom
最近在研究一个有趣的排序问题。如图所示,通过2个栈S1和S2,Tom
希望借助以下4种操作实现将输入序列升序排序。
操作a
如果输入序列不为空,将第一个元素压入栈S1
操作b
如果栈S1不为空,将S1栈顶元素弹出至输出序列
操作c
如果输入序列不为空,将第一个元素压入栈S2
操作d
如果栈S2不为空,将S2栈顶元素弹出至输出序列
如果一个1−n的排列P可以通过一系列操作使得输出序列为1,2,…,(n−1),Tom
就称P是一个“可双栈排序排列”。例如(1,3,2,4)就是一个“可双栈排序序列”,而(2,3,4,1)不是。下图描述了一个将(1,3,2,4)排序的操作序列:<a,c,c,b,a,d,d,b>
当然,这样的操作序列有可能有几个,对于上例(1,3,2,4),<a,c,c,b,a,d,d,b>是另外一个可行的操作序列。Tom
希望知道其中字典序最小的操作序列是什么。
输入格式
第一行是一个整数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
思路
一个简单的递推算法。
这个题目其实不需要二分图,也不需要搜索。
只要思考如果能够排序,元素需要满足什么性质,然后贪心地向A栈添加即可。
代码
#include<cmath> #include<cstdio> #include<cstring> #include<iostream> #include<algorithm> using namespace std; const int N=1010; char ans[2*N]; int in[N],a[N],b[N]; int ad,bd,k,n,cnt,asd; bool check(int k) { if(!bd) return 1; int i,j; for(i=k+1; i<=n; i++) if(in[i]>in[k]&&in[i]>b[bd]) break; for(j=i+1; j<=n; j++) if(in[j]<in[k]) return false; return true; } int main () { bool able=1; int at=1; scanf("%d",&n); a[0]=b[0]=1e4,asd=1; for(int i=1; i<=n; i++) scanf("%d",&in[i]); for(int i=1; i<=(n<<1); i++) { if(a[ad]==asd) { ad--; asd++; ans[++cnt]='b'; continue; } if(b[bd]==asd) { bd--; asd++; ans[++cnt]='d'; continue; } if(at<=n&&in[at]<a[ad]&&check(at)) { a[++ad]=in[at]; ans[++cnt]='a'; at++; continue; } if(at<=n&&in[at]<b[bd]) { b[++bd]=in[at]; ans[++cnt]='c'; at++; continue; } able=0; break; } if(able) for(int i=1; i<=cnt; i++) putchar(ans[i]),putchar(' '); else printf("0"); printf("\n"); return 0; }