利用线性表解决问题(停车场管理)

#include <stdio.h>

#include <malloc.h>

#define N 3                    /*停车场内最多的停车数*/

#define M 3                    /*候车场内最多的停车数*/

#define Price 2             /*每单位时间停车费用*/

typedef struct

{

    int CarNo[N];           /*车牌号*/

    int CarTime[N];         /*进场时间*/

    int top;                /*栈指针*/

} SqStack;                  /*定义顺序栈类型,用于描述停车场*/

typedef struct

{

    int CarNo[M];           /*车牌号*/

    int front,rear;         /*队首和队尾指针*/

} SqQueue;                  /*定义循环队类型,用于描述候车场*/

//顺序栈

void InitStack(SqStack *&s)

{

    s=(SqStack *)malloc(sizeof(SqStack));

    s->top=-1;

}

int StackEmpty(SqStack *s)

{  return(s->top==-1);

}

int StackFull(SqStack *s)

{

    return(s->top==N-1);

}

void Push(SqStack *&s,int e1,int e2)

{

    if (s->top==N-1)

    printf("栈已满\n");

    else{

s->top++;

    s->CarNo[s->top]=e1;

    s->CarTime[s->top]=e2;}

}

int Pop(SqStack *&s,int &e1,int &e2)

{

    if (s->top==-1) {printf("处于栈底,无法删除\n");return 0;}

    else{

e1=s->CarNo[s->top];

    e2=s->CarTime[s->top];

    s->top--;

    return 1;}

}

void DispStack(SqStack *s)

{

    int i;

    for (i=s->top; i>=0; i--)

        printf("%d ",s->CarNo[i]);

    printf("\n");

}

//循环队列

void InitQueue(SqQueue *q)

{

    q=(SqQueue *)malloc (sizeof(SqQueue));

   

    q->front=q->rear=NULL;

}

int QueueEmpty(SqQueue *q)

{

    return(q->front==q->rear);

}

int QueueFull(SqQueue *q)       

{

    return ((q->rear+1)%M==q->front);

}

int enQueue(SqQueue *q,int e)      /*入队*/

{

    if ((q->rear+1)%M==q->front)    /*队满*/

        return 0;

    q->rear=(q->rear+1)%M;

    q->CarNo[q->rear]=e;

    return 1;

}

int deQueue(SqQueue *q,int &e)     /*出队*/

{

    if (q->front==q->rear)          /*队空*/

        return 0;

    q->front=(q->front+1)%M;

    e=q->CarNo[q->front];

    return 1;

}

void DispQueue(SqQueue *q)      /*输出队中元素*/

{

    int i;

    i=(q->front+1)%M;

    printf("%d ",q->CarNo[i]);

    while ((q->rear-i+M)%M>0)

    {

        i=(i+1)%M;

        printf("%d ",q->CarNo[i]);

    }

    printf("\n");

}

int main()

{

    int order;

    int no,e1,time,e2;

    int i,j,t;

    SqStack *St,*St1;  //St是停车场,St1是在有车离开时,记录为该车移开位置的车辆

    SqQueue *Qu;   //Qu是候车场

    InitStack(St);

    InitStack(St1);

    InitQueue(Qu);

    do

    {

        printf("输入指令(1:到达 2:离开 3:显示停车场 4:显示候车场 0:退出):");

        scanf("%d",&order);

        switch(order)

        {

        case 1:     /*汽车到达*/

            printf("输入车号和时间(设车号和时间均为整数): ");

            scanf("%d %d",&no,&time);

            if (!StackFull(St))         /*停车场不满*/

            {

                Push(St,no,time);

                printf("  >>停车场位置:%d\n",St->top+1);

            }

            else                        /*停车场满*/

            {

                if (!QueueFull(Qu))     /*候车场不满*/

                {

                    enQueue(Qu,no);

                    printf("  >>候车场位置:%d\n",Qu->rear);

                }

                else

                    printf("  >>候车场已满,不能停车\n");

            }

            break;

        case 2:     /*汽车离开*/

            printf("输入车号和时间(设车号和时间均为整数): ");

            scanf("%d%d",&no,&time);

            for (i=0; i<=St->top && St->CarNo[i]!=no; i++);  //在栈中找

            if (i>St->top)

                printf("  >>未找到该编号的汽车\n");

            else

            {

                t = St->top - i;  //需要出栈的车辆数目

                for (j=0; j<t; j++)  //for (j=i; j<=St->top; j++)

                {

                    Pop(St,e1,e2);

                    Push(St1,e1,e2);        /*倒车到临时栈St1中*/

                }

                Pop(St,e1,e2);              /*该汽车离开*/

                printf("  >>%d汽车停车费用:%d\n",no,(time-e2)*Price);

                while (!StackEmpty(St1))    /*将临时栈St1重新回到St中*/

                {

                    Pop(St1,e1,e2);

                    Push(St,e1,e2);

                }

                if (!QueueEmpty(Qu))        /*队不空时,将队头进栈St*/

                {

                    deQueue(Qu,e1);

                    Push(St,e1,time);       /*以当前时间开始计费*/

                }

            }

            break;

        case 3:     /*显示停车场情况*/

            if (!StackEmpty(St))

            {

                printf("  >>停车场中的车辆:"); /*输出停车场中的车辆*/

                DispStack(St);

            }

            else

                printf("  >>停车场中无车辆\n");

            break;

        case 4:     /*显示候车场情况*/

            if (!QueueEmpty(Qu))

            {

                printf("  >>候车场中的车辆:"); /*输出候车场中的车辆*/

                DispQueue(Qu);

            }

            else

                printf("  >>候车场中无车辆\n");

            break;

        case 0:     /*结束*/

            if (!StackEmpty(St))

            {

                printf("  >>停车场中的车辆:"); /*输出停车场中的车辆*/

                DispStack(St);

            }

            if (!QueueEmpty(Qu))

            {

                printf("  >>候车场中的车辆:"); /*输出候车场中的车辆*/

                DispQueue(Qu);

            }

            break;

        default:    /*其他情况*/

            printf("  >>输入的命令错误\n");

            break;

        }

    }

    while(order!=0);

    return 0;

}

停车场模拟管理程序的设计与实现 1.设计目的 理解线性表的逻辑结构和存储结构,进一步提高使用理论知识指导解决实际问题的能力。 2.问题描述 设停车场只有一个可停放几辆汽车的狭长通道,只有一个大门可供汽车进出。汽车在停车场内按车辆到达的先后顺序依次排列,若车场内已停满几辆汽车,则后来的汽车只能在门外的便道上等候,一旦停车场内有车辆开走,则排在便道上的第一辆汽车即可进入;当停车场内某辆汽车要开走时,由于停车场是狭长的通道,在它之后开入的车辆必须先推出车场为他让路,待车辆开出大门,为他让路的车辆再按原次序进入车场。试设计这样一个停车场模拟管理程序。 3.数据结构设计 (1)为了便于区分每辆汽车并了解每辆车当前所处的位置,需要记录汽车的牌照号码和汽车当前的状态。 (2)为了便于停车场管理,要为每个车位分配一个固定的编号。 (3)当停车场的停车位上都已停满了汽车,又有新的汽车到来时要把它调度到便道上,便道上的车辆要按照进入便道的先后顺序顺次序放在便道上,为便道上的每个位置分配一个固定的编号。当有车从停车位上离开后,便道上的第一辆汽车就立即进入停车位上的某个车位。 (4)当某辆车离开停车场的时候,比他后进停车位的车要为他让路,而且当他开走之后让路的车还要按照原来的停放次序再次进入停车位的某个车位上,为完成这项功能,定义一个结构体。 4.功能(函数)设计 本程序从总体上分为四个功能模块,分别为: (1)程序功能介绍和操作提示模块 (2)汽车进入停车位的管理模块 (3)汽车离开停车位的管理模块 (4)查看停车场状态的查询模块 5.界面设计 6.编码实现 7.运行与测试 (1)连续有7辆汽车到来,牌照号分别为CF001、CF002、CF003、CF004、CF005、CF006、CF007,前5辆车应该进入停车位1-5车位,第6、7辆车应停入便道的1、2位置上。 (2)上面(1)中的情况发生后,让牌照CF003的汽车从停车场开走,应显示CF005、CF004的让路动作和CF006从便道到停车位的动作。 (3)随时检查停车位和便道的状态,不应该出现有空位而便道上还有车的情况。 (4)其它正常操作的一般情况。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值