停车场管理模拟程序

2024-03-13 09:32

本文主要是介绍停车场管理模拟程序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

#include<iostream>
#include<fstream>
using namespace std;

#define MAX_SIZE  2//停车场能够容纳的车的数量
#define FARE     5//表示停车场的收费为每小时FARE元

int CountForStack=0;// 此全局变量用来计数堆栈现有的车数
int CountForQueue=0;// 此全局变量用来计数队列现有的车数

typedef struct//这个节点用来保存每辆车的信息
{
 char Condition ;//用来表示“到达”或者“离开”的状态," "表示还没有到达,也没有离开
 int  ArriveTime;//用来记录到达时间,默认为-1,说明还没有到达
 int  LeaveTime;// 用来记录离开时间,默认为-1,说明还没有离开
 int  License;// 记录车牌号
}CarNode;

typedef struct //栈的定义
{
 CarNode *base;//栈底指针,指向0
 CarNode *top;//栈顶指针,如果指向0,说明栈为空
 int stacksize;//栈的容量大小
}CarStack;

typedef struct QNode//队列节点的定义
{
 char Condition ;//用来表示“到达”或者“离开”的状态," "表示还没有到达,也没有离开
 int  ArriveTime;//用来记录到达时间,默认为-1,说明还没有到达
 int  LeaveTime;// 用来记录离开时间,默认为-1,说明还没有离开
 int  License;// 记录车牌号
 QNode *next;//指向下一个节点的指针

}QNode;

typedef struct// 队列的定义
{
 QNode *front;//队头指针
 QNode *rear;//队尾指针
}Queue;


bool InitStack(CarStack &S)//此函数用来初始化栈
{
 S.base =(CarNode *)malloc(MAX_SIZE*sizeof(CarNode));
 if(!S.base )
 {
  cout<<"内存分配失败!"<<endl;
  return false;//说明内存分配失败,返回false
 }

 S.top =S.base ;
 S.stacksize =MAX_SIZE;
 return true;
}

bool InitQueue(Queue &Q)//此函数用来初始化队列
{
 Q.front =(QNode *)malloc(sizeof(QNode));
 if(!Q.front )
 {
  cout<<"内存分配失败!"<<endl;
  return false;
 }
 Q.rear =Q.front ;
 Q.front ->next =0;//下一个节点指空
 return true;
}

bool EnQueue(Queue &Q,QNode &qnode)//此函数用来入队一个节点
{
 QNode *p=(QNode *)malloc(sizeof(QNode));
 if(!p)
 {
  cout<<"内存分配失败!"<<endl;
  return false;
 }

 p->ArriveTime =qnode.ArriveTime ;
 p->Condition =qnode.Condition ;
 p->LeaveTime =qnode.LeaveTime ;
 p->License =qnode.License ;
 p->next =0;

 Q.rear ->next =p;
 Q.rear =p;
 return true;
}

bool DeQueue(Queue &Q,QNode &t)//此函数用来出队
{
 if(Q.front ==Q.rear )
 {
  cout<<"队列为空!"<<endl;
  return false;
 }

 QNode *p=Q.front->next ;
    t.ArriveTime =p->ArriveTime ;
 t.Condition =p->Condition ;
 t.LeaveTime =p->LeaveTime ;
 t.License =p->License ;
 Q.front ->next =p->next ;
 if(Q.rear ==p)//如果P是指向最后一个出队的元素
  Q.rear =Q.front ;
 free(p);
 return true;
}

 

void InitCarNode(CarNode &C,char condition,int arrivetime,int leavetime,int license)//本函数用来初始化一个CarNode 节点
{
 C.ArriveTime =arrivetime;
 C.Condition =condition;
 C.LeaveTime =leavetime;
 C.License=license;
}
 


bool Push(CarStack &S,CarNode &car)//此函数用来入栈一个CarNode 节点
{
 if(S.top -S.base >=S.stacksize )
 {
  cout<<"此栈已满,不能压入新的信息"<<endl;
  return  false;
 }

 (*S.top ).ArriveTime =car.ArriveTime ;
 (*S.top ).Condition =car.Condition ;
 (*S.top ).LeaveTime =car.LeaveTime ;
 (*S.top ).License =car.License ;
 ++S.top ;//栈顶指针上移
 return true;
}

bool Pop(CarStack &S,CarNode &t)//此函数用来弹出栈内元素
{
 if(S.top ==S.base )
 {
  cout<<"栈空,不能执行出栈操作!"<<endl;
  return false;
 }

    --S.top ;//栈顶指针下移
 t.ArriveTime =(*S.top ).ArriveTime ;
 t.Condition =(*S.top ).Condition ;
 t.LeaveTime =(*S.top ).LeaveTime ;
 t.License =(*S.top ).License ;

 return  true;
}

bool IsStackFull(CarStack &S)//此函数用来判断堆栈是否已满
{
 if(S.top -S.base >=S.stacksize )
  return true;
 else 
  return false;
}

bool IsStackEmputy(CarStack &S)//此函数用来判断堆栈是否为空
{
 if(S.top ==S.base )
  return true;
 else 
  return false;
}

bool IsQueueEmputy(Queue &Q)//此函数用来判断队列是否为空
{
 if(Q.front ==Q.rear )
  return true;
 else
  return false;
}


bool SearchInStack(CarStack&S,int a)//a表示要查找的车牌号,如果在停车场里面,就返回true
{  
 bool tag=false;
 if(!IsStackEmputy(S))
 {
 CarNode *p=S.top-1 ;
 while(p!=S.base )
 {
  if((*p).License ==a)
   tag=true;
  --p;
 }
 if((*p).License ==a)
   tag=true;
 }
 return tag;
}

bool SearchInQueue(Queue &Q,int a)//a表示要查找的车牌号,如果在通道里面,就返回true
{
 bool tag=false;
 if(!IsQueueEmputy(Q))//如果队列非空
 {
  QNode *p=Q.front->next ;
  while(p!=Q.rear)
  {
   if((*p).License ==a)
    tag=true;
  }//退出此while循环时p指向最后一个元素
     if((*p).License ==a)
    tag=true;
 }
 return tag;
}


  


void InCar(CarStack &S,Queue &Q,int a1,int a2)//此函数用来表示进入车辆, 参数a1用来表示到达时间,参数a2表示车牌号码
{
    if(SearchInStack(S,a2))
 {
  cout<<"车号"<<a2<<"已经存在于停车场内,输入有误"<<endl;
  return ;
 }

 if(SearchInQueue(Q,a2))
 {
        cout<<"车号"<<a2<<"已经存在于通道内,输入有误"<<endl;
  return ;
 }


 if(IsStackFull(S))//如果堆栈已满,说明停车场已满,需要停车在通道
 {
  QNode  qnode;
  qnode.ArriveTime =-1;//在通道时间不收费,所以不计时
  qnode.Condition ='A';
  qnode.LeaveTime =-1;//定义为-1,说明还没有开始离开
  qnode.License =a2;
  EnQueue(Q,qnode);//停在通道上
  ++CountForQueue;
  cout<<"车号:"<<qnode.License <<"停在通道的第"<<CountForQueue<<"号位置"<<endl;
 }

 else
 {
  CarNode carnode;
  carnode.ArriveTime =a1;
  carnode.Condition ='A';
  carnode.LeaveTime =-1;
  carnode.License =a2;
  Push(S,carnode);
  ++CountForStack;
        cout<<"车号:"<<carnode.License <<"到达时间 "<<carnode.ArriveTime <<"停在停车场的第"<<CountForStack<<"号位置"<<endl;
 }
}

void OutCar(CarStack &S,Queue &Q,int a1,int a2)//此函数用来出车,参数a1用来表示离开时间,参数a2表示车牌号码
{    
 if(SearchInQueue(Q,a2))
 {
        cout<<"车号"<<a2<<"存在于通道内,还未进入停车场,不能离开"<<endl;
  return ;
 }
    if(!SearchInStack(S,a2))
 {
  cout<<"车号"<<a2<<"不在停车场内,输入有误"<<endl;
  return ;
 }

 CarStack tempstack;
 InitStack(tempstack);//建立并且初始化用于暂存出车时让车的堆栈

    bool  tag1=false;//标志这个停车场出车以前是否已满,默认为没有满
 tag1=IsStackFull(S);
 bool  tag2=true;//标志通道是否有汽车在等待, 默认为通道为空
 tag2=IsQueueEmputy(Q);

 CarNode temp;//用来保存暂时取出的汽车
 bool tag3=false;//用来标志是否是离开时间小于到达时间而导致离开失败,true表示离开失败
 while(1)//让车离开
 {
  Pop(S,temp);
  if(temp.License ==a2)
  {  
   if(a1<temp.ArriveTime )
   
   {
    cout<<"输入有误,离开时间不能小于到达时间,离开失败"<<endl;
    tag3=true;
    Push(tempstack,temp);
   }
   else
   cout<<"车号"<<a2<<"现在离开停车场,所用的时间为"<<a1-temp.ArriveTime <<"收费为"<<(a1-temp.ArriveTime )*FARE<<endl;
   break;
  }
  else
   Push(tempstack ,temp);//进入暂存栈
 }
    
 while(!IsStackEmputy(tempstack))//倒出的车再次进入停车场
 {
  Pop(tempstack,temp);
  Push(S,temp);
 }
   
 QNode tempqnode;//用来暂时保存从通道出来的汽车
 if(tag1==true&&tag2==false&&tag3==false)//如果出车前停车场已满,并且通道不为空,并且离开没有失败
 {
  DeQueue(Q,tempqnode);
  --CountForQueue;
  temp.ArriveTime =a1 ;
  temp.Condition =tempqnode.Condition ;
  temp.LeaveTime =tempqnode.LeaveTime ;
  temp.License =tempqnode.License ;
  Push(S,temp);
 }
 if(tag2==true&&tag3==false)// 如果停车场没有满,并且离开成功
  --CountForStack;

}

void showmenu(CarStack &S,Queue &Q)
{
 cout<<"******************************选择菜单*******************************"<<endl;
 cout<<"                     1: 停车"<<endl;
 cout<<"                     2: 离开"<<endl;
 cout<<"                     3: 退出"<<endl;
 cout<<"*****************************请按键选择******************************"<<endl;
 int tag;
 cin>>tag;
 while(tag!=1&&tag!=2&&tag!=3)
  cin>>tag;
 int a1;
 unsigned int a2;
 switch(tag)
 {
 case 1:

  cout<<"请输入到达的车号"<<endl;
  cin>>a1;
  cout<<"请输入到达时间"<<endl;
  cin>>a2;
  InCar(S,Q,a2,a1);
  break;
  
 case 2:
  cout<<"请输入离开的车号"<<endl;
  cin>>a1;
  cout<<"请输入离开的时间"<<endl;
  cin>>a2;
  OutCar(S,Q,a2,a1);
  break;

 case 3:
  return ;
  break;
 }
   
 char ch;
 cout<<"*******************按E/e退出,按任意键返回菜单**************************"<<endl;
 cin>>ch;
 if(ch!='E'&&ch!='e')
  showmenu(S,Q);
}

 

 

void main()
{
 CarStack carstack;
 InitStack(carstack);// 建立并且初始化用于停车场的堆栈

 Queue   carqueue;
 InitQueue(carqueue);//建立并且初始化用于通道的队列

 showmenu(carstack,carqueue);


}
 

这篇关于停车场管理模拟程序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/804405

相关文章

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

Redis实现高效内存管理的示例代码

《Redis实现高效内存管理的示例代码》Redis内存管理是其核心功能之一,为了高效地利用内存,Redis采用了多种技术和策略,如优化的数据结构、内存分配策略、内存回收、数据压缩等,下面就来详细的介绍... 目录1. 内存分配策略jemalloc 的使用2. 数据压缩和编码ziplist示例代码3. 优化的

SpringBoot集成XXL-JOB实现任务管理全流程

《SpringBoot集成XXL-JOB实现任务管理全流程》XXL-JOB是一款轻量级分布式任务调度平台,功能丰富、界面简洁、易于扩展,本文介绍如何通过SpringBoot项目,使用RestTempl... 目录一、前言二、项目结构简述三、Maven 依赖四、Controller 代码详解五、Service

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

Linux系统管理与进程任务管理方式

《Linux系统管理与进程任务管理方式》本文系统讲解Linux管理核心技能,涵盖引导流程、服务控制(Systemd与GRUB2)、进程管理(前台/后台运行、工具使用)、计划任务(at/cron)及常用... 目录引言一、linux系统引导过程与服务控制1.1 系统引导的五个关键阶段1.2 GRUB2的进化优

Spring Security 前后端分离场景下的会话并发管理

《SpringSecurity前后端分离场景下的会话并发管理》本文介绍了在前后端分离架构下实现SpringSecurity会话并发管理的问题,传统Web开发中只需简单配置sessionManage... 目录背景分析传统 web 开发中的 sessionManagement 入口ConcurrentSess

Linux之UDP和TCP报头管理方式

《Linux之UDP和TCP报头管理方式》文章系统讲解了传输层协议UDP与TCP的核心区别:UDP无连接、不可靠,适合实时传输(如视频),通过端口号标识应用;TCP有连接、可靠,通过确认应答、序号、窗... 目录一、关于端口号1.1 端口号的理解1.2 端口号范围的划分1.3 认识知名端口号1.4 一个进程

SpringBoot结合Knife4j进行API分组授权管理配置详解

《SpringBoot结合Knife4j进行API分组授权管理配置详解》在现代的微服务架构中,API文档和授权管理是不可或缺的一部分,本文将介绍如何在SpringBoot应用中集成Knife4j,并进... 目录环境准备配置 Swagger配置 Swagger OpenAPI自定义 Swagger UI 底

Linux权限管理与ACL访问控制详解

《Linux权限管理与ACL访问控制详解》Linux权限管理涵盖基本rwx权限(通过chmod设置)、特殊权限(SUID/SGID/StickyBit)及ACL精细授权,由umask决定默认权限,需合... 目录一、基本权限概述1. 基本权限与数字对应关系二、权限管理命令(chmod)1. 字符模式语法2.

在macOS上安装jenv管理JDK版本的详细步骤

《在macOS上安装jenv管理JDK版本的详细步骤》jEnv是一个命令行工具,正如它的官网所宣称的那样,它是来让你忘记怎么配置JAVA_HOME环境变量的神队友,:本文主要介绍在macOS上安装... 目录前言安装 jenv添加 JDK 版本到 jenv切换 JDK 版本总结前言China编程在开发 Java