【火车出栈】ZOJ - 2603 Railroad Sort

好久没写递归了,怕手生再来练练手。

题意:车轨上有上图所示的n个中转栈,现有2n个列车,给出列车初始编号序列。列车从最右边驶入车轨,并且列车只能从右向左移动,要求给出列车中转操作序列,使列车经过这n个中转栈后从最左端输出时满足1~2n的排列。

分析:其实这不就是火车出栈问题吗,可以用分治递归的思路,首先使一半有序,然后再合并使整体有序。

思路:具体来讲就是,一段无序的序列(编号1~2n),通过一个中转栈使1 ~ 2n-1在整个序列的前面2n-1+1 ~ 2n在整个序列的后面;再经过下一中转栈时,又能进一步进行排序,这样在最后一个中转栈是就只是两两之间的有序无序问题了。

什么都不如例子来的好使,比如n=3时,初始序列为 2 1 5 8 7 3 6 4

经过栈1时,使序列满足如下形式:2 1 3 4 | 6 7 8 5

经过栈2时,使序列满足如下形式:2 1 | 3 4 | 6 5 | 8 7

经过栈3时,使序列满足如下形式:1 2 | 3 4 | 5 6 | 7 8

然后就达到要求了。

那么该如何操作才能达到上述状态呢?

比如经过栈1时,可将序列分成包含(1~4)和(5~8)的两段子序列。那么每来一个数,判断该数是否大于4,若小于等于4,则该列车进栈接着出栈;否则该列车暂时停放在栈中,等所有小于等于4的列车出栈后再按照先进后出的性质出栈,这样就达到了目的;

经过栈1时的操作序列:2 2 1 1 5 8 7 3 3 6 4 4 6 7 8 5

经过下面的栈的情况就类似了,不再赘述了。

附代码:

 1 #include <iostream>
 2 #include <cstring>
 3 #include <cstdlib>
 4 #include <cstdio>
 5 #include <stack>
 6 using namespace std;
 7 const int maxn = 5000;
 8 int a[maxn];
 9 void dfs(int num[], int r, int k)
10 {
11     if(k <= 1)  return ;
12     int s1[k/2 + 1];
13     int s2[k/2 + 1];
14     int k1 = 0, k2 = k/2;
15     int l = r-k;
16     int mid = (l + r)/2;
17
18     stack<int> si;
19     int p1, p2; p1 = p2 = 0;
20     for(int i = 0; i < k; i++)
21     {
22         if(num[i] <= mid)
23         {
24             s1[p1++] = num[i];
25             printf("%d %d ", num[i], num[i]);
26         }
27         else
28         {
29             printf("%d ", num[i]);
30             si.push(num[i]);
31         }
32     }
33     while(!si.empty())
34     {
35         s2[p2] = si.top();
36         printf("%d ", s2[p2++]);
37         si.pop();
38     }
39     dfs(s1, mid, k/2);
40     dfs(s2, r, k/2);
41     return ;
42 }
43
44
45 int main()
46 {
47     int n;
48     while(~scanf("%d", &n) && n)
49     {
50         int m = 1<<n;
51         for(int i = 0; i < m; i++)
52             scanf("%d", &a[i]);
53         dfs(a, m, m);
54         printf("\n");
55     }
56     return 0;
57 }

时间: 2024-11-03 22:31:22

【火车出栈】ZOJ - 2603 Railroad Sort的相关文章

全是套路——出栈顺序

实习终于结束了,好久没有总结编程问题了,因为最近真是没怎么写程序. 今天做华为的火车进站问题,这是一道看似像全排列,其实是栈的问题. 看了一些网上的写法,掌握了一种思路,也是种投机取巧的方法: 先产生序列的全排列,然后再全排列里面去除不可能的出栈顺序. 1.全排列很好产生,用stl里面的next_permutation(begin,end) 2.判断这个排列是不是正确的出栈顺序. 就第2点: 1.给一种进栈顺序加上序号,比如4,2,8这么进栈,则4为0,2为1,8为2,以此类推.只是想表明一个顺

给定入栈序列,求出合法的出栈序列的个数

思想:1.利用全排列函数next_permutation()求出所有可能的序列 2.从中选出所有正确的序列 #include <iostream> #include <stack> #include <vector> #include <algorithm> using namespace std; //判断序列是否是合法的出栈序列 bool IsPopOrder(const int* pPush,const int* pPop,int n) { if (p

栈与队列问题1——出栈序列

问题描述:栈是常用的一种数据结构,有n个元素在栈顶端一侧等待进栈,栈顶端另一侧是出栈序列.你已经知道栈的操作有两种:push和pop,前者是将一个元素进栈,后者是将栈顶元素弹出.现在要使用这两种操作,由一个操作序列可以得到一系列的输出序列.请你编程求出对于给定的n,计算并输出由操作数序列1,2,…,n,经过一系列操作可能得到的输出序列总数. 分析:之前就有看过这种问题.就是火车进站问题,判断序列是否合法,当时是用STL栈做的.这个题只需统计次数,那么,方法就十分简便了,递归和动归都可以实现,当然

出栈顺序 与 卡特兰数(Catalan)的关系

一,问题描述 给定一个以字符串形式表示的入栈序列,请求出一共有多少种可能的出栈顺序?如何输出所有可能的出栈序列? 比如入栈序列为:1 2 3  ,则出栈序列一共有五种,分别如下:1 2 3.1 3 2.2 1 3.2 3 1.3 2 1 二,问题分析 先介绍几个规律: ①对于出栈序列中的每一个数字,在它后面的.比它小的所有数字,一定是按递减顺序排列的. 比如入栈顺序为:1 2 3 4. 出栈顺序:4 3 2 1是合法的,对于数字 4 而言,比它小的后面的数字是:3 2 1,且这个顺序是递减顺序.

PAT线性结构_一元多项式求导、按给定步长反转链表、出栈序列存在性判断

02-线性结构1. 一元多项式求导 (25) 设计函数求一元多项式的导数.(注:xn(n为整数)的一阶导数为n*xn-1.) 输入格式:以指数递降方式输入多项式非零项系数和指数(绝对值均为不超过1000的整数).数字间以空格分隔. 输出格式:以与输入相同的格式输出导数多项式非零项的系数和指数.数字间以空格分隔,但结尾不能有多余空格.注意“零多项式”的指数和系数都是0,但是表示为“0 0”. 输入样例: 3 4 -5 2 6 1 -2 0 输出样例: 12 3 -10 1 6 0 最简单的方式是用

POJ1363 Rails 验证出栈序列问题(转)

题目地址: http://poj.org/problem?id=1363 此题只需验证是否为合法的出栈序列. 有两个思路:1.每个已出栈之后的数且小于此数的数都必须按降序排列.复杂度O(n^2),适合人脑. //思路 1 不对!!! 例如 数据 ,               3 5 2 4 1              --------                正确答案为 no 2.另一个思路就是直接模拟入栈出栈过程.虽然模拟毫无技巧可言,但复杂度O(n),优于算法1.适合电脑. 代码如

火车进出栈 java

题目描述 一列火车n节车厢,依次编号为1,2,3,-,n.每节车厢有两种运动方式,进栈与出栈,问n节车厢出栈的可能排列方式有多少种. 输入 一个数,n(n<=60000) 输出 一个数s表示n节车厢出栈的可能排列方式 题解: 这题要用大数,java 然后卡特兰数:有个数学模型 s[i]=c(n,2n)/(n+1) 组合数公式:这个总是记不住 这个题不能直接求,会TLE.需要先求出结果的每个质因数有多少次幂,然后用快速幂求,再把所有求幂得到的结果乘起来. http://www.cnblogs.co

火车出站

火车出站 题目描述 铁路进行列车调度时,常把站台设计成栈式结构的站台,试问:设有编号为1到n的n辆列车,顺序开入栈式结构的站台,则可能的出栈序列有多少种? 输入 输入包含多组测试数据.每组为一个正整数n(1<=n<=20),表示有n辆列车. 输出 输出可能的出栈序列有多少种. 样例输入 4 3 样例输出 14 5 1 #include<iostream> 2 using namespace std; 3 int main() 4 { 5 int n; 6 while(cin>

不要搜索,出栈序列统计

1627: 出栈序列统计 时间限制: 1 Sec  内存限制: 128 MB 题目描述 栈是常用的一种数据结构,有n令元素在栈顶端一侧等待进栈,栈顶端另一侧是出栈序列.你已经知道栈的操作有两·种:push和pop,前者是将一个元素进栈,后者是将栈顶元素弹出.现在要使用这两种操作,由一个操作序列可以得到一系列的输出序列.请你编程求出对于给定的n,计算并输出由操作数序列1,2,-,n,经过一系列操作可能得到的输出序列总数. 输入 一个整数n(1<=n<=15) 输出 一个整数,即可能输出序列的总数