1.2 出栈序列统计

【问题描述】

栈是常用的一种数据结构,有n令元素在栈顶端一侧等待进栈,栈顶端另一侧是出栈序列。你已经知道栈的操作有两·种:pushpop,前者是将一个元素进栈,后者是将栈顶元素弹出。现在要使用这两种操作,由一个操作序列可以得到一系列的输出序列。请你编程求出对于给定的n,计算并输出由操作数序列12,…,n,经过一系列操作可能得到的输出序列总数。

【输入】

一个整数n1<=n<=15

【输出】

一个整数,即可能输出序列的总数目。

【样例】

stack1.in stack1.out

35

【算法分析】

此题最原始的方法是利用回溯,对栈操作的模拟,边界是

       1.所有的元素全部出栈

        2.所有的元素经过操作

回溯法pascal代码:

var
   n,sum:longint;
procedure dfs(top,head:longint);{top为栈顶指针,head为已经有多少个元素入过栈}
var
   i,t:longint;
begin
  if  head=n then
    begin
      inc(sum);
      exit;
    end;
  if top>0 then dfs(top-1,head);
  if head<n+1 then dfs(top+1,head+1);
end;
begin
  readln(n);
  sum:=0;
  dfs(0,0);
  writeln(sum);
end.

        回溯法理解简单,但是效率奇差,n在比较小的情况下能够胜任,但是n一旦达到比较大的数时,必定超时!那有没有效率更高的算法呢?

           回溯法效率差的原因是记录了很多的重复运算,其实此题还可以用动归方法解决,f[i,j],i表示入栈的个数,j表示出栈的个数,那f[i,j]就表示入栈i个数中出j个数的,但是此题要注意的是出栈数不能大于入栈数,那动归方程该如何推导,再次谢谢一位某位具有探索精神的大神为我们做了细致的研究现在我把它的论文粘贴如下:

       

1      引 言

在实际应用和数据结构课程的教学中,栈作为一种基本结构非常重要[1][3][4][6]。已知给定序列,求出栈序列的数目、求所有出栈的序列、以及判断某个序列是否为合法的出栈序列[5][7],这类问题经常出现。在[3]中,对出栈序列的计数问题给出了介绍性的说明,由于结果的证明需要用到生成函数,[3]也只是直接给出了结论。本文提出使用“两点之间路径计数的方法解决出栈序列的计数问题,在此基础上可以求所有出栈的序列,以及判断某个序列是否为合法的出栈序列。

2      问题分析

           两点之间路径计数的问题

问题:假设AB两点之间所有道路如图1中的方形网格线(5×5),规定从AB只能够向右或向上移动,求A点到B点有几条路。


两点之间路径计数问题的路径图

分析:因为规定从AB只能够向右或向上移动,因此任意一点只能从该点的左邻点或下邻点到达,例如,任意一点C点只能从D点或E点到达。因此,从A点到C点的路只能由这两部分组成:①从A点到D点,再从D点到C点;②从A点到E点,再从E点到C点。

结论1:A点到任意一点C的路径数目=A点到D点(C的左邻点)的路径数目+A点到E点(C的下邻点)的路径数目

其中,由于A点正上方的点没有左邻点,而且问题中已规定从AB只能够向右或向上运动,所以A点到A点正上方点的路径数目为1。同理,A点到A点正右方点的路径数目为1。

根据结论1,将A点到任意一点的路径数目求出,如图1中网格线交点处的数字所示。

           栈的操作与两点之间路径计数问题的操作的比较

栈的操作有两种:入栈、出栈。其中需要注意的问题有三个:①所有节点入栈之后只能出栈;②栈空时只能入栈;③其它情况下入栈、出栈任意执行。

两点之间路径计数问题的操作有两种:向右移动、向上移动。其中需要注意的问题有三个:①移到最右边后只能向上移;②移到最上边只能向右移;③其它情况下上移、右移任意执行。

由以上分析可见,栈的操作与两点之间路径计数问题的操作有很大的相似性,不妨将入栈和右移相关联,出栈和上移相关联。但是,这样关联之后,由于两个问题并不等价(例如,图1中的D点,按照栈的操作是不可到达的),所以需要对图1中的所有点进一步分析。

           对操作关联后图1中点的分析

首先,在图1中添加A点到B点的对角虚线。这条虚线将所有的点分成三类:①虚线上的点;②虚线左上方的点;③虚线右下方的点。

其次,按照两点之间路径计数问题规定的操作容易得出:①从A点移到虚线上每一个点时,所执行的右移操作次数和上移操作次数相等;②从A点移到虚线左上方每一个点时,所执行的右移操作次数小于上移操作次数;③从A点移到虚线右下方每一个点时,所执行的右移操作次数大于上移操作次数。

再次,由于栈操作过程中的任意时刻必须有:入栈操作次数≥出栈操作次数(取等号时栈空)。

很明显,图1中虚线左上方的点按照栈的操作是不可到达的,虚线上的点恰好是栈空时的状态,虚线右下方的点按照栈的操作都可以到达,所以考虑修改图1中的路径图。

           改进后的路径图及规则

将图1中虚线左上方的点去掉后如图2所示(5×5方形网格线的下三角)。


改造后的的路径图

规定:AB只能够向右或向上移动,右移为入栈操作,上移为出栈操作。

根据2.3的分析可得结论2

  从A点到A点正右方的点的路径数目 = 1;

  从A点到每一行最左的点(考虑B点,不考虑A点)的路径数目 =从A点到该点的下邻点的路径数目;

  A点到其它任意一点C的路径数目=A点到D点(C的左邻点)的路径数目+A点到E点(C的下邻点)的路径数目;

  按照栈的操作从A点开始到B点,图2中的所有点都是可到达的;

  4个节点的入栈、出栈操作完全包含在图2中;

  将⑤扩展得:N个节点的出栈、入栈操作完全包含在(N+1)×(N+1)方形网格线的下三角中。

在此仅对结论2第⑤点作一些说明:首先A点和对角线上的其它点表示栈空,只能入栈(右移);其次,移到最右的竖边时所有的元素都已经入栈,只能出栈(上移);再次,B点为最终状态,不能入栈也不能出栈;最后,其它的点可以任意入栈(右移)、出栈(上移)。所以4个节点的入栈、出栈操作完全包含在图2中。

从结论2中可以看到栈的操作与两点之间路径计数问题的操作在图2中是等价的。根据结论2,将A点到任意一点的路径数目求出,如图2中网格线交点处的数字所示,图2中虚线箭头表示了执行结论2第①点,图2中实线箭头表示了执行结论2第③点。

           结论2推广

将结论2加以推广得结论3对于如图2的形式((N+1)×(N+1)方形网格线的下三角),规定AB只能够向右或向上移动,右移为入栈操作,上移为出栈操作,所求A点到B点的路径数目就是N个节点出栈序列的数目,并且从A点到B点的每一条路都代表一种出栈序列。

3      设计实现

求N个节点出栈序列数目的算法在具体实现时,采用由下向上逐行处理,每一行从左至右逐点处理的方法。此外,在 计算当前行的值时,只需要使用上一行的值;在计算各行中的每一个值时左邻点的值为数组中当前元素的前一个元素,下邻点为数组中当前元素,把数组中当前元素 的前一个元素加到当前元素上就求出了当前点的值,所以在具体实现时,只使用一个数组来保存当前行的值即可。由于最后一行只有一个数,也是该行的第一个数, 根据结论2第③点可知该数在倒数第二行中已经计算出来,所以该行不用计算,直接取上一行计算结果中的最后一个数即可。

4      结论

该方法简单方便,不需要记忆任何公式[3],特别适合没有组合数学基础的人员。另外,根据图2还可以设计算法将入栈、出栈的操作序列求出来,这样就可以得到所有的出栈序列。同时根据图2也可以判断某个序列是否为合法的出栈序列,可以解决[5][7]中车厢调度问题。

  为表感谢,特把该作者文章标红显示,通过该大神深入浅出的讲解,我相信只要不是比我还笨的人都能理解了,大神为了方便讲解所以把向右定义为入栈,向上定义为出栈,但为了方便书写代码,我把向下定义为出栈,向右定义为入栈,那么动归方程就推导如下f[i,j]:=f[i-1,j]+f[i,j-1],pascal代码如下:

var
  n,tot:longint;
  i,j:longint;
  f:array[0..100,0..100] of longint;
begin
  readln(n);
  fillchar(f,sizeof(f),0);
  for i:=1 to n do f[i,0]:=1;{上边界全为1}
  for i:=1 to n do
    for j:=1 to i do
      begin
        if j<i then f[i,j]:=f[i-1,j]+f[i,j-1];
        if i=j then f[i,j]:=f[i,j-1];
      end;
  writeln(f[n,n]);
end.

      动归比回溯效率从阶乘的效率提高到n^2的效率,那此题还能继续优化吗,那是当然,那就是公式,据说此题是一个经典的卡特兰数,对此我深信不疑,但是鉴于水学水平有限至今不知如何证明,但有人有排列组合还是能理解一二,但是还是有点模糊,不管了,先把别人的成果贴出来,等以后慢慢再理解吧


  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值