\(Sol\)

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

再来考虑双栈排序,如果把这个序列划分为两个子序列,这两个子序列分别可以单栈排序,那么这个序列就可以双栈排序了.

怎么划分子序列呢?预处理一个后缀最小值\(f_i=min_{j=i}^na_j\),若\(i<j\)且\(f_{j+1}<a_i<a_j\),那么说明\(i,j\)两个点是不能共存的.因为\(k\)在\(i\)后面,\(i\)只能入栈,但\(j\)也要入栈,而\(a_i<a_j\),不能使得较小的\(i\)先出栈.我们现在得到了若干对关系\((i,j)\)表示它们不能共存,现在我们要求把序列分成两部分满足任意一个部分内部没有这样的关系.这不就是二分图染色嘛\(QwQ\).染色完了之后就直接分别按照单栈的做就行.单栈做的时候还要注意字典序尽量小.

还有\(lx\)的优秀贪心做法,大概讲下:从第一个开始扫,一个数能进入\(stack1\)的条件是不能出现下面这种情况:在后面的数中有比该数和\(stack2\)的栈顶元素都大的元素,在这个大的元素的后面又有一个比该数小的数.简单理解下,因为后面有一个比该数小的数,所以该数入栈后不能在大元素要入栈前清出,大元素也不能进入\(stack2\),因为\(stack2\)的栈顶要等比它小的该数清出它才能清出,所以这种情况是不合法的.

\(Code\)

#include<bits/stdc++.h>
#define il inline
#define Ri register int
#define go(i,a,b) for(Ri i=a;i<=b;++i)
#define yes(i,a,b) for(Ri i=a;i>=b;--i)
#define e(i,u) for(Ri i=b[u];i;i=a[i].nt)
using namespace std;
il int read()
{
Ri x=0,y=1;char c=getchar();
while(c<'0'||c>'9'){if(c=='-')y=-1;c=getchar();}
while(c>='0'&&c<='9'){x=(x<<1)+(x<<3)+c-'0';c=getchar();}
return x*y;
}
const int N=1001;
int n,p[N],mi[N],b[N],ct,s1[N],s2[N],t1,t2;bool col[N],vis[N];
struct nd{int v,nt;}a[N*2];
il void add(Ri u,Ri v){a[++ct]=(nd){v,b[u]};b[u]=ct;}
il bool dfs(Ri u,bool co)
{
col[u]=co;vis[u]=1;
e(i,u)
{
Ri v=a[i].v;
if(!vis[v])if(!dfs(v,1-co))return 0;;
if(vis[v] && col[v]!=1-co)return 0;
}
return 1;
}
il void ins1(Ri x){printf("a ");s1[++t1]=x;}
il void ins2(Ri x){printf("c ");s2[++t2]=x;}
il void pus1(){printf("b ");--t1;}
il void pus2(){printf("d ");--t2;}
int main()
{
n=read();
go(i,1,n)p[i]=read();
mi[n+1]=1e9;yes(i,n,1)mi[i]=min(mi[i+1],p[i]);
go(i,1,n)
go(j,i+1,n)
if(p[i]<p[j] && p[i]>mi[j+1])add(i,j),add(j,i);
go(i,1,n)
if(!vis[i])if(!dfs(i,0)){puts("0");return 0;};
go(i,1,n)
{
if(col[i]==0)
if(s1[t1]>p[i])ins1(p[i]);
else {while(t1 && s1[t1]<p[i])pus1();;ins1(p[i]);}
else
{
while(s1[t1]<mi[i+1])pus1();
if(s2[t2]>p[i])ins2(p[i]);
else {while(t2 && s2[t1]<p[i])pus2();;ins2(p[i]);}
}
}
while(t1 || t2)
{
if(!t2){pus1();continue;}
if(!t1){pus2();continue;}
if(s1[t1]<s2[t2])pus1();
else pus2();
}
return 0;
}

随机推荐

  1. js+canvas 一只一担小游戏

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  2. H3C IP的主要作用

  3. 阿里云ECS服务器活动99元一年,最高可买三年

    这几天阿里云 99一年.279三年的服务器活动如火如荼,和之前腾讯三年的服务器非常类似,非常低的价格换取非常高的价值,当然,通常情况下便宜没好货的,想要玩一下的老铁可以进阿里云去看看,阿里云270三年 ...

  4. div+css布局的好处

    改版的时候更方便,只需改动CSS文件 页面加载速度更快,结构清晰,页面简洁 表现与结构相分离 有利于SEO,排名更靠前

  5. Python--day25--接口类多继承

  6. 2019-9-2-Visual-Studio-自定义项目模板

    title author date CreateTime categories Visual Studio 自定义项目模板 lindexi 2019-09-02 12:57:38 +0800 2018 ...

  7. P1073 奇数还是偶数

    题目描述 给你 \(N(1 \le N \le 1000)\) 个数,你需要判断每个数是奇数还是偶数. 输入格式 输入的第一行包含一个整数 \(N(1 \le N \le 1000)\) . 接下来 ...

  8. P1039 大规模间谍入侵

    题目描述 爱丽丝魔法王国成立10周年,于是决定矩形国庆大阅兵. 在国庆大阅兵期间,为了防止暗黑王国的间谍乔装成平民混入,需要对每一个进城的人做检测. 因为暗黑王国的人长得和爱丽丝魔法王国的人长得很像, ...

  9. SpringBoot 上传文件到linux服务器 异常java.io.FileNotFoundException: /tmp/tomcat.50898……解决方案

    SpringBoot 上传文件到linux服务器报错java.io.FileNotFoundException: /tmp/tomcat.50898-- 报错原因: 解决方法 java.io.IOEx ...

  10. win10 uwp 好看的时间选择控件

    本文告诉大家我找到的好看的时间选择控件 先给大家看一下图,然后就知道我说的是什么 首先需要安装 Nuget ,搜索 DeanChalk.UWP.TimePicker 或输入Install-Packag ...