程序设计要求用C语言写一个简单的火车售票系统,主要实现的功能为:
录入班次信息
浏览班次信息
按班次号查询
按终点站查询
按余票数量排序保存
售票
退票
更新班次信息
退出系统
所有的班次信息保存在number.dat文件中,排序过后的保存在sort.dat中(.dat是一种二进制文件)。
在编写的过程中我觉得在判断火车的状态比较值得深究。这里假设火车主要有四种状态:
1.未发车
2.已发车
3.停止检票
4.停止退票
在程序中,思路是将代表发车时间的字符串转化为整型,再和系统现在的时间进行大小比较,主要采用if判断各种情况。其中atime代表的是发车时间的整型数,btime代表的是系统时间的整型数,具体实现如下:
1
2
3
4
5
6
7
|
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之内,此时代表停止退票。
判断检票和上面思想大致相同。
为了参考研究,贴上代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
|
#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(); } } |
程序中肯定还存在很多问题,自己对调用系统函数方面还有很多需要学习,文件的操作也不是很熟练,欢迎大家一起探讨交流。