基于C语言实现简单的12306火车售票系统

 更新时间:2016年09月08日 10:45:23   作者:lcr_happy  
火车售票系统给我们的出行带来了极大的方面,那么他基于编程是如何实现的呢?今天小编抽时间给大家分享一个使用C语言写的一个简单的火车票系统,感兴趣的朋友参考下

程序设计要求用C语言写一个简单的火车售票系统,主要实现的功能为:

录入班次信息

浏览班次信息

按班次号查询

按终点站查询

按余票数量排序保存

售票

退票

更新班次信息

退出系统

所有的班次信息保存在number.dat文件中,排序过后的保存在sort.dat中(.dat是一种二进制文件)。

在编写的过程中我觉得在判断火车的状态比较值得深究。这里假设火车主要有四种状态:

1.未发车

2.已发车

3.停止检票

4.停止退票

在程序中,思路是将代表发车时间的字符串转化为整型,再和系统现在的时间进行大小比较,主要采用if判断各种情况。其中atime代表的是发车时间的整型数,btime代表的是系统时间的整型数,具体实现如下:

if(atime<=btime) //已经发车
return 1;
if(((atime-btime<=30)&&(atime-btime>5)&&(atime/100==btime/100))||(((atime%100+(60-btime%100))<=30)&&(atime%100+(60-btime%100))>5&&(atime/100-btime/100==1))) //距发车半小时以内,停止退票,%表示取余
return 2;
if(((atime-btime<=5)&&(atime/100==btime/100))||((atime%100+(60-btime%100)&&(atime/100-btime/100==1))<=5)) //距发车前五分钟内停止检票
return 3;
return 0; //可以办理购退票

在判断退票时如果两个时间的小时数是一样的,则它们的分钟数如果相差在30之内即半小时之内或者发车时间的小时和系统时间的小时相差一个1,并且发车时间的分钟数小于30,系统时间的分钟大于30,则它们之间也就相差在30之内,此时代表停止退票。

判断检票和上面思想大致相同。

为了参考研究,贴上代码:

#include<stdio.h>
#include<stdlib.h>
#include<conio.h>
#include<string.h>
#include "time.h"
//班次信息表
#define SIZELIMIT 10 //每班次的具体信息的字符个数限制在10以内
#define MAXNUM 1000 //设定最多只能录入1000个车次信息
typedef struct cardbase //定义一个关于班次信息的结构体,取名为CardBase
{
char C_ID[SIZELIMIT];//班次
char C_TIME[SIZELIMIT];//发车时间
char C_ANAME[SIZELIMIT];//起点站
char C_BNAME[SIZELIMIT];//终点站
char C_USETIME[SIZELIMIT];//行车时间
char C_MAXNUM[SIZELIMIT];//额定载量
char C_REMAINNUM[SIZELIMIT];//余票数量
}CardBase;
int cbNum=0;//记录班次数
CardBase cBList[MAXNUM];//班次列表
//读取班次信息
void readcardbasefile()
{
FILE *fp;
//打开文件失败则创建文件
if((fp=fopen("./number.dat","r"))==NULL)
{
//初次运行创建文件
if ((fp=fopen("./number.dat","w"))==NULL)
{
exit(0); //返回
}
else
{
fclose(fp);
}
return ;
}
/*文件位置指针移动到文件末尾*/
fseek(fp,0,2);//重定位文件内部位置指针
/* int fseek( FILE *stream, long offset, int origin );
第一个参数stream为文件指针
第二个参数offset为偏移量,正数表示正向偏移,负数表示负向偏移
第三个参数origin设定从文件的哪里开始偏移,可能取值为:SEEK_CUR、 SEEK_END 或 SEEK_SET
SEEK_SET: 文件开头
SEEK_CUR: 当前位置
SEEK_END: 文件结尾
其中SEEK_SET,SEEK_CUR和SEEK_END依次为0,1和2.
简言之:
fseek(fp,100L,0);把stream指针移动到离文件开头100字节处;
fseek(fp,100L,1);把stream指针移动到离文件当前位置100字节处;
fseek(fp,-100L,2);把stream指针退回到离文件结尾100字节处。 */
if (ftell(fp)>0)//文件不为空
/*ftell函数用于得到文件位置指针当前位置相对于文件首的偏移字节数
也就是得到文件所包含的字节数,如果大于0,则代表文件非空*/
{
//文件位置指针移动到文件开始
rewind(fp);
char buff[10]={0};
for (cbNum=0;!feof(fp) && fread(&cBList[cbNum],sizeof(CardBase),1,fp);cbNum++)
/*对于feof函数,如果遇到文件结束,函数feof(fp)的值为非零值,否则为0。
即如果文件结束,!feof(fp)为0,跳出循环
对于fread函数,是从一个文件流中读数据,如果调用成功返回实际读取到的项个数,
如果不成功或读到文件末尾返回 0*/
fgets(buff,10,fp);
/*char *fgets(char *buf, int bufsize, FILE *stream);
从文件结构体指针stream中读取数据,每次读取一行。读取的数据保存在buf指向的字符数组中,
每次最多读取bufsize-1个字符(第bufsize个字符赋'\0'),如果文件中的该行,不足bufsize个字符,
则读完该行就结束。如若该行(包括最后一个换行符)的字符数超过bufsize-1,则fgets只返回一个不完整的行,
但是,缓冲区总是以NULL字符结尾,对fgets的下一次调用会继续读该行。函数成功将返回buf,失败或读到文件结尾返回NULL。
因此我们不能直接通过fgets的返回值来判断函数是否是出错而终止的,应该借助feof函数或者ferror函数来判断。
*/
fclose(fp);
}
else
{
fclose(fp);
}
return;
}
//保存班次信息
void writecardbasefile()
{
int i;
FILE *fp;
if ((fp=fopen("./number.dat","w"))==NULL)
{
printf("系统错误");
}
char buff[10]={0};
strcpy(buff,"\r\n");
for (i=0;i<cbNum;i++)
{
if (fwrite(&cBList[i],sizeof(CardBase),1,fp)!=1)
{
printf("系统错误");
}
if (fwrite(buff,2,1,fp)!=1)
{
printf("系统错误");
}
}
fclose(fp);
}
//保存排序信息
void writesortfile()
{
int i;
FILE *fp;
if ((fp=fopen("./sort.dat","w"))==NULL)
{
printf("系统错误");
}
char buff[10]={0};
strcpy(buff,"\r\n");
for (i=0;i<cbNum;i++)
{
if (fwrite(&cBList[i],sizeof(CardBase),1,fp)!=1)
{
printf("系统错误");
}
if (fwrite(buff,2,1,fp)!=1)
{
printf("系统错误");
}
}
fclose(fp);
}
//打印并输入后返回
void printReturn(char *info)
{
printf("\n\n\t %s",info);
fflush(stdin);//清空输入缓冲区,通常是为了确保不影响后面的数据读取.
getchar();
}
//输入信息
void setInfo(char pinfo[1024],char desinfo[])
{
printf("\n\t%s:",pinfo);
fflush(stdin);
scanf("%s",desinfo);
}
//系统初始化
void initsystem()
{
readcardbasefile();
};
//录入班次
void infoinput()
{
setInfo("班次",cBList[cbNum].C_ID);
setInfo("发车时间(24小时制)",cBList[cbNum].C_TIME);
setInfo("起点站",cBList[cbNum].C_ANAME);
setInfo("终点站",cBList[cbNum].C_BNAME);
setInfo("行车时间",cBList[cbNum].C_USETIME);
setInfo("额定载量",cBList[cbNum].C_MAXNUM);
setInfo("余票数量",cBList[cbNum].C_REMAINNUM);
cbNum++;
writecardbasefile();
printReturn("\n\t录入成功,回车键返回");
};
//时间比较
int timecmp(char A[10])
{
//将火车时间转成整数
char tempa[10]={0};
int ta=0;
int i;
for(i=0;i<strlen(A);i++)
if(A[i]!=':'&&A[i]!=':') //冒号的中文输入和英文输入
{
tempa[ta]=A[i];
ta++;
}
int atime=atoi(tempa);//把字符串变成整型
//获取系统时间
char tempb[10]={0};
time_t t = time(0);
strftime( tempb, 10, "%H%M",localtime(&t) );
int btime=atoi(tempb);
//比较
if(atime<=btime) //已经发车
return 1;
if(((atime-btime<=30)&&(atime-btime>5)&&(atime/100==btime/100))||(((atime%100+(60-btime%100))<=30)&&(atime%100+(60-btime%100))>5&&(atime/100-btime/100==1))) //距发车半小时以内,停止退票,%表示取余
return 2;
if(((atime-btime<=5)&&(atime/100==btime/100))||((atime%100+(60-btime%100)&&(atime/100-btime/100==1))<=5)) //距发车前五分钟内停止检票
return 3;
return 0; //可以办理购退票
}
//浏览所有班次
void queryallinfo()
{
printf("班次信息\n");
printf("班次 发车时间 起点站 终点站 行车时间 额定载量 余票数量 状态\n");
int i;
for(i=0;i<cbNum;i++)
{
char temp[20]={0};
strcpy(temp,"未发车");
if(1==timecmp(cBList[i].C_TIME))
strcpy(temp,"已发车");
if(2==timecmp(cBList[i].C_TIME))
strcpy(temp,"停止退票");
if(3==timecmp(cBList[i].C_TIME))
strcpy(temp,"停止检票");
printf("%-010s%-010s%-010s%-010s%-010s%-010s%-010s%s\n",
cBList[i].C_ID,cBList[i].C_TIME,
cBList[i].C_ANAME,cBList[i].C_BNAME,cBList[i].C_USETIME,
cBList[i].C_MAXNUM,cBList[i].C_REMAINNUM,temp );
}
printReturn("\n\t回车键返回");
};
//通过班次号查路线
void queryinfobyID()
{
char ID[20]={0};
setInfo("输入班次号",ID);
int i;
for( i=0;i<cbNum;i++)
{
if(strcmp(cBList[i].C_ID,ID)==0)
{
printf("班次信息\n");
printf("班次 发车时间 起点站 终点站 行车时间 额定载量 余票数量\n");
printf("%-010s%-010s%-010s%-010s%-010s%-010s%-010s\n",
cBList[i].C_ID,cBList[i].C_TIME,
cBList[i].C_ANAME,cBList[i].C_BNAME,cBList[i].C_USETIME,
cBList[i].C_MAXNUM,cBList[i].C_REMAINNUM );
printReturn("\n\t回车键返回");
return;
}
}
printReturn("\n\t指定信息不存在,回车键返回");
};
//通过终点站查路线
void queryinfobyBNAME()
{
char Name[20]={0};
setInfo("输入终点站",Name);
int i;
for(i=0;i<cbNum;i++)
{
if(strcmp(cBList[i].C_BNAME,Name)==0)
{
printf("班次信息\n");
printf("班次 发车时间 起点站 终点站 行车时间 额定载量 余票数量\n");
printf("%-010s%-010s%-010s%-010s%-010s%-010s%-010s\n",
cBList[i].C_ID,cBList[i].C_TIME,
cBList[i].C_ANAME,cBList[i].C_BNAME,cBList[i].C_USETIME,
cBList[i].C_MAXNUM,cBList[i].C_REMAINNUM );
printReturn("\n\t回车键返回");
return;
}
}
printReturn("\n\t指定信息不存在,回车键返回");
};
//排序按余票数量排序保存
void sortSave()
{
//冒泡排序
int i,j;
for(i=0;i<cbNum;i++)
for(j=0;j<cbNum-i-1;j++)
{
if(atoi(cBList[j].C_REMAINNUM)<atoi(cBList[j+1].C_REMAINNUM))
{
CardBase temp=cBList[j];
cBList[j]=cBList[j+1];
cBList[j+1]=temp;
}
}
//打印排序结果
queryallinfo();
//保存排序结果
writesortfile();
};
//售票
void sale()
{
char ID[20]={0};
setInfo("输入班次号",ID);
int i;
for(i=0;i<cbNum;i++)
{
if(strcmp(cBList[i].C_ID,ID)==0)
{
if(cBList[i].C_REMAINNUM==0)
{
printReturn("\n\t余票不足,回车返回");
return;
}
//余票减少
int temp=atoi(cBList[i].C_REMAINNUM)-1;
if(temp<0) temp=0;//确保余票不为负数
_itoa(temp,cBList[i].C_REMAINNUM,10);
//保存到文件
writecardbasefile();
printf("班次信息\n");
printf("班次 发车时间 起点站 终点站 行车时间 额定载量 余票数量\n");
printf("%-010s%-010s%-010s%-010s%-010s%-010s%-010s\n",
cBList[i].C_ID,cBList[i].C_TIME,
cBList[i].C_ANAME,cBList[i].C_BNAME,cBList[i].C_USETIME,
cBList[i].C_MAXNUM,cBList[i].C_REMAINNUM );
printReturn("\n\t售票成功,回车返回");
return;
}
}
printReturn("\n\t指定班次不存在,回车键返回");
};
//退票
void back()
{
char ID[20]={0};
setInfo("输入班次号",ID);
int i;
for(i=0;i<cbNum;i++)
{
if(strcmp(cBList[i].C_ID,ID)==0)
{
//余票增加
int temp=atoi(cBList[i].C_REMAINNUM)+1;
_itoa(temp,cBList[i].C_REMAINNUM,10);
//保存到文件
writecardbasefile();
printf("班次信息\n");
printf("班次 发车时间 起点站 终点站 行车时间 额定载量 余票数量\n");
printf("%-010s%-010s%-010s%-010s%-010s%-010s%-010s\n",
cBList[i].C_ID,cBList[i].C_TIME,
cBList[i].C_ANAME,cBList[i].C_BNAME,cBList[i].C_USETIME,
cBList[i].C_MAXNUM,cBList[i].C_REMAINNUM );
printReturn("\n\t\t退票成功,回车返回");
return;
}
}
printReturn("\n\t指定班次不存在,回车键返回");
};
//更新车次信息
void update()
{
int flag;
char ID[20]={0};
setInfo("请输入要删除或修改的车次(请确认车次输入正确)",ID);
do
{
printf("\n\t按下列提示更新车次信息(退出修改请按0):");
printf("\n\t1:删除车次;2:修改发车时间;3:修改起点站;4:修改终点站;");
printf("\n\t5:修改行车时间;6:修改额定载量;7:修改余票数量;0:退出修改;");
printf("\n\t请选择:");
scanf("%d",&flag);
int i;
CardBase temp;
for(i=0;i<cbNum;i++)
{
if(strcmp(cBList[i].C_ID,ID)==0)
{
if(flag==1) //删除车次信息
{
int j;
for(j=i;j<cbNum;j++)
cBList[j]=cBList[j+1];//后面的数据覆盖前面的数据
cbNum--;//车次数减一
//保存到文件
writecardbasefile();
}
if(flag==2) //修改发车时间
{
setInfo("请输入新的发车时间",cBList[i].C_TIME);
temp=cBList[i];
//保存到文件
writecardbasefile();
}
if(flag==3) //修改起点站
{
setInfo("请输入新的起点站",cBList[i].C_ANAME);
temp=cBList[i];
//保存到文件
writecardbasefile();
}
if(flag==4) //修改终点站
{
setInfo("请输入新的终点站",cBList[i].C_BNAME);
temp=cBList[i];
//保存到文件
writecardbasefile();
}
if(flag==5) //修改行车时间
{
setInfo("请输入新的行车时间",cBList[i].C_USETIME);
temp=cBList[i];
//保存到文件
writecardbasefile();
}
if(flag==6) //修改额定载量
{
setInfo("请输入新的额定载量",cBList[i].C_MAXNUM);
temp=cBList[i];
//保存到文件
writecardbasefile();
}
if(flag==7) //修改余票数量
{
setInfo("请输入新的余票数量",cBList[i].C_REMAINNUM);
temp=cBList[i];
//保存到文件
writecardbasefile();
}
}
}
//更改完成后显示更新后的班次信息,如果是删除车次则不显示,即flag不等于0的时候显示更新信息
if(flag!=1&&flag!=0)
{ printf("更新后的班次信息\n");
printf("班次 发车时间 起点站 终点站 行车时间 额定载量 余票数量\n");
printf("%-010s%-010s%-010s%-010s%-010s%-010s%-010s\n",
temp.C_ID,temp.C_TIME,
temp.C_ANAME,temp.C_BNAME,temp.C_USETIME,
temp.C_MAXNUM,temp.C_REMAINNUM );
}
}while(flag);
printReturn("\n\t完成车次信息更新,回车键返回");
};
void mainmenu()
{
while(1)
{
char select;
do{
system("cls");//清屏
printf("\n\t ╭═════════■□■□═══╮");
printf("\n\t│ 火车班次系统 │");
printf("\n\t╰═══■□■□══════════╯");
printf("\n\t ┌────────────────┐");
printf("\n\t │1.录入班次 │");
printf("\n\t │2.浏览所有班次 │");
printf("\n\t │3.通过班次号查路线 │");
printf("\n\t │4.通过终点站查路线 │");
printf("\n\t │5.排序保存 │");
printf("\n\t │6.售票 │");
printf("\n\t │7.退票 │");
printf("\n\t │8.更新车次信息 │");
printf("\n\t │0.退出登录 │");
printf("\n\t └────────────────┘");
printf("\n\t 请选择:");
fflush(stdin);//清空输入缓冲区,通常是为了确保不影响后面的数据读取.
select=getchar();//等待用户输入数据
}while (select<'0'||select>'8');
switch(select)
{
case '0':exit(0);break;
case '1':infoinput();break;
case '2':queryallinfo();break;
case '3':queryinfobyID();break;
case '4':queryinfobyBNAME();break;
case '5':sortSave();break;
case '6':sale();break;
case '7':back();break;
case '8':update();break;
}
}
}
//主函数
int main()
{
initsystem(); //系统初始化
while(1)
{
mainmenu();
}
}

程序中肯定还存在很多问题,自己对调用系统函数方面还有很多需要学习,文件的操作也不是很熟练,欢迎大家一起探讨交流。

相关文章

  • c++项目构成从cmake使用基础详解

    c++项目构成从cmake使用基础详解

    这篇文章主要为大家介绍了c++项目构成,从cmake使用基础开始为大家讲解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-11-11
  • c++程序字符型的实例讲解

    c++程序字符型的实例讲解

    在本篇文章里小编给大家整理的是一篇关于
    2020-02-02
  • C++ pair的用法案例详解

    C++ pair的用法案例详解

    这篇文章主要介绍了C++ pair的用法案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-09-09
  • C语言 野指针与空指针专篇解读

    C语言 野指针与空指针专篇解读

    全网最接地气的C语言野指针介绍,此处对于野指针与空指针知识点做一些简要的介绍,作者实属初学,写博客也是作者学习的一个过程,难免文章中有内容理解不到位或者有不当之处,还请朋友们不吝指正,希望大家多多给予支持,赠人玫瑰,手有余香
    2021-11-11
  • C++超集C++/CLI模块的基本语法

    C++超集C++/CLI模块的基本语法

    这篇文章介绍了C++超集C++/CLI模块的基本语法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-07-07
  • C语言矩阵连乘 (动态规划)详解

    C语言矩阵连乘 (动态规划)详解

    这篇文章主要介绍了C语言矩阵连乘 (动态规划)详解的相关资料,需要的朋友可以参考下
    2017-05-05
  • 浅谈返回函数内部new分配的内存的引用

    浅谈返回函数内部new分配的内存的引用

    下面小编就为大家带来一篇浅谈返回函数内部new分配的内存的引用。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-12-12
  • 一文带你了解C语言中的0长度数组(可变数组/柔性数组)

    一文带你了解C语言中的0长度数组(可变数组/柔性数组)

    众所周知, GNU/GCC 在标准的 C/C++ 基础上做了有实用性的扩展, 零长度数组(Arrays of Length Zero) 就是其中一个知名的扩展,本文就来聊聊零长度数组的相关知识吧
    2023-03-03
  • 一个string类的简单实现案例

    一个string类的简单实现案例

    下面小编就为大家带来一篇一个string类的简单实现案例。小编觉得挺不错的现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-01-01
  • C++递归算法处理岛屿问题详解

    C++递归算法处理岛屿问题详解

    这篇文章主要介绍了用递归算法解决岛屿问题的流程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
    2022-10-10

最新评论