数据结构实验之栈与队列七:出栈序列判定

Problem Description
给一个初始的入栈序列,其次序即为元素的入栈次序,栈顶元素可以随时出栈,每个元素只能入栈依次。输入一个入栈序列,后面依次输入多个序列,请判断这些序列是否为所给入栈序列合法的出栈序列。

例如序列1,2,3,4,5是某栈的压入顺序,序列4,5,3,2,1是该压栈序列对应的一个出栈序列,但4,3,5,1,2就不可能是该序列的出栈序列。假设压入栈的所有数字均不相等。

Input
第一行输入整数n(1<=n<=10000),表示序列的长度。

第二行输入n个整数,表示栈的压入顺序。

第三行输入整数t(1<=t<=10)。

后面依次输入t行,每行n个整数,表示要判断的每一个出栈序列。

Output
对应每个测试案例输出一行,如果由初始入栈序列可以得到该出栈序列,则输出yes,否则输出no。
题解:
根据栈有后进先出的特点,出栈的序列无非有两种情况:刚进栈接着出栈,进栈待一会儿再出栈。如果一个序列全部都是刚进栈接着出栈那么出栈序列与原序列相同,如果有元素在栈内待了一会,出栈序列就与原序列不同。例如题给的例子原序列为12345,出栈序列为45321,说明123进栈并在栈里待着,然后4进栈接着出栈,然后5进栈接着出栈,最后123再出栈,顺序是321,连起来就是45321。
在判断的时候可以分别用两个数组存储原序列和需要判断的序列,然后从头开始比较两个序列,如果两个序列的元素相同,说明这些元素是刚进栈就出栈了,是合理的,如果碰到两个序列不同的元素,就先让原序列的元素进栈,直到比较到这两个序列有元素相同或者原序列结束,再让栈里的元素依次出栈,再与需要比较的序列比较,如果一致就是合理的出栈序列,否则不是。

Sample Input
5
1 2 3 4 5
2
4 5 3 2 1
4 3 5 1 2
Sample Output
yes
no

#include <stdio.h>
#include <stdlib.h>
#define stackinitsize 11000
#define stackcreat 10
#define ok 1
#define error -1
#define overflow -1
typedef int elemtype;
typedef struct
{
    elemtype *base;
    elemtype *top;
    int stacksize;

} sqstack;
int initstack(sqstack *s)
{
    s->base=(elemtype*)malloc(stackinitsize*sizeof(elemtype));
    if(!s->base)exit(overflow);
    s->top=s->base;
    s->stacksize=stackinitsize;
    return ok;
}
int push(sqstack *s,elemtype e)
{
    if(s->top-s->base>=s->stacksize)
    {
        s->top=(elemtype*)realloc(s->base,(s->stacksize+stackcreat)*sizeof(elemtype));
        if(!s->base)exit(overflow);
        s->top=s->base+s->stacksize;


    }*(s->top++)=e;
    return ok;

}
int pop(sqstack *s)
{
    if(s->top==s->base)return error;
    s->top--;
    return ok;

}
int gettop(sqstack *s)
{
    int e;
    if(s->top==s->base)return error;
    e=*(s->top-1);
    return e;

}
int stackempty(sqstack *s)
{
    if(s->top==s->base)return 1;
    else return 0;

}
int main()
{
    int t,n,i,a[11000],b[11000],j,e;
    sqstack s;
    scanf("%d",&n);
    for(i=1; i<=n; i++)scanf("%d",&a[i]);//储存原序列
    scanf("%d",&t);
    while(t--)
    {
        for(i=1; i<=n; i++)scanf("%d",&b[i]);//储存需要比较的序列
        initstack(&s);
        i=j=1;
        int f=1;
        while(i<=n)
        {
            if(a[i]==b[j])//如果两个序列的元素相同,是合法的
            {
                i++;
                j++;

            }
            else
            {
                push(&s,a[i]);//如果不相同,让原序列的元素进栈
                i++;

            }

        }
        while(!stackempty(&s))//让栈内的元素出栈
        {
            e=gettop(&s);
            if(e==b[j])//如果相同,则合法
            {
                j++;
                pop(&s);


            }
            else
            {
                f=0;
                break;

            }

        }
        if(f==0)printf("no\n");
        else printf("yes\n");

    }

    return 0;
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值