【问题描述】
有一个火车站,铁路如图所示,每辆火车从A驶入,再从B方向驶出,同时它的车厢可以重新组合。假设从A方向驶来的火车有n节(n<=1000),分别按照顺序编号为1,2,3,…,n。假定在进入车站前,每节车厢之间都不是连着的,并且它们可以自行移动到B处的铁轨上。另外假定车站C可以停放任意多节车厢。但是一旦进入车站C,它就不能再回到A方向的铁轨上了,并且一旦当它进入B方向的铁轨,它就不能再回到车站C。 负责车厢调度的工作人员需要知道能否使它以a1,a2,…,an的顺序从B方向驶出,请来判断能否得到指定的车厢顺序。
【输入】
输入文件的第一行为一个整数n,其中n<=1000,表示有n节车厢,第二行为n个数字,表示指定的车厢顺序。
【输出】
如果可以得到指定的车厢顺序,则输出一个字符串 ”YES”,否则输出”NO” 。
【输入样例】
5 5 4 3 2 1
【输出样例】
YES

【解法一】
常规思路:(3 1 2)不可行的序列形式,当且仅当存在 ai>ak>aj (i<j<k)

【解法二】
分析: 车站C相当于一个栈。我们用模拟法来做,假设我们已经处理了前i-1节从B方向驶出的车厢,我们现在要让ai驶出。若ai不在车站C中,我们就让若干车厢从A方向驶入车站C,直到ai驶入,再将它从B方向驶出;若ai在车站C中,如果它是车站C中停在最前面的,则将它从B方向驶出,否则原问题无解。 如样例中,出栈序列是3 5 4 2 1,模拟过程如下: ①一开始栈为空 ②由于3不在栈中,就需要把1,2,3依次进栈,再出栈,这样符合出栈序列第一个数是3,当前栈为{1,2} ③第2个出栈的是5,5不在栈中,则就把4,5压栈,再出栈就可以得到5,此时栈为{1,2,4} ④第3个出栈的是4,正好是栈顶元素,直接出栈,栈变为{1,2} ⑤第4个出栈的是2,正好是栈顶元素,直接出栈,栈变为{2} ⑥第5个出栈的是1,正好是栈顶元素,直接出栈,栈变为{} 在模拟过程中没有碰到要出栈的数在栈中但不是栈顶元素的情况,所以该方案可行。
1 #include <iostream> 2 #include <algorithm> 3 #include <cstring> 4 using namespace std; 5 const int N = 1010; 6 int stack[N],a[N]; int top,n; 7 int main() 8 { cin >> n; 9 for (int i = 1;i <= n;++ i) 10 cin >> a[i]; 11 top = 0; 12 for (int i = 1,cur = 1;i <= n;++ i) //cur为当前要从A方向驶入的车厢号 13 { 14 while (cur <= a[i]) 15 stack[++ top] = cur ++; 16 if (stack[top] == a[i]) 17 -- top; 18 else 19 { 20 cout << "NO" << endl; 21 return 0; 22 } 23 } 24 cout << "YES" << endl; 25 return 0; 26 }