这其实是学校小学期的课题,拿来分享一下设计的过程蛤。划分一下水平,本人处于大二。这篇设计经历适用于那些出入C语言基础一段时间的娃。
航空订票系统是个较为复杂的系统,我负责其中管理员一块,写管理员登陆系统以及订单管理,个人信息管理一块。下面是我从提交论文筛选出来的部分。
Part 0-系统总体框架
然后是具体细则
Part 1-管理员登录系统
登录系统需要解决的重点主要有这么几个问题:字符串匹配,密码和账户名绑定的问题。下面具体描述码代码的过程,问题也会顺便解决阐述。
管理员输入账户和密码,分别存放在两个已经预留足够空间大小的字符串中。为了说明简便,先将账户放在声明过的su_account[81]中,密码放在su_password[81]中。
printf("ID:");
gets(su_account); //接受一个字符串,放在su_account中
printf("Password:");
gets(su_password);//接受一个字符串,放在su_password中
这里没有添加容错机制,因为管理员账户不需要通过注册系统来添加,是固定的。
在输入账户和密码之后,系统就需要开始进行验证账户确定性以及身份。简单原理是字符串匹配:
首先和账户匹配,然后再和密码匹配。就是说,先在一个存放账户和密码的txt文档里找到账户,然后在找是否有这个密码,找到了就成功了。但事实上会存在一个bug:我输入的账户和001管理员匹配上了,但我输入的密码是002的,这种情况下,上面的简单匹配是识别不出来的。所以需要一个手段将账户和密码对应起来,防止错乱匹配的情况发生。
所以,这时候使用结构体数据类型是一个很好的办法。
具体是这样操作的:
在编译时会声明一个结构体数组,这个数组能够存放足够的管理员帐户信息条目。每个结果结构体数组元素有:管理员帐户,密码,公司,姓名,性别,邮箱。
这样就可以将密码和同一账户绑定起来了。只要先检索账户,账户名匹配成功后,在同一个结构体中匹配密码,这样就不会出现错乱匹配。
小功能
Html网页输入密码时,往往不会在密码栏显示明文密码,而是显示实心圆
事实上C语言可以做到类似的密码输入,而且很简单。
关键只需要掌握getch()函数。这个函数会接受一个字符,但不显示在屏幕上。
所以简单描述功能具体实现:设置一个存放密码的字符串(预留足够空间),使用循环,每使用getch接受一个字符,就将字符存放在字符串的一个元素里,然后字符数组偏移量加一,继续使用getch存放在下一个元素里。同时,每输入一个字符(循环),就在屏幕上输出"*",以此类推,知道输入一个回车键(转义字符\r)。基本功能就是这样。
Part2 简单接口菜单
登录过后应该是进入管理员系统,之前简单写过具体功能实现。在众多功能下,需要一个总体菜单,通往各个具体的功能。这个菜单非常容易,只需要一个system函数就好。
system函数只需要在引号里放入文件路径就好,其实system也有其他的功能,简单列出几个:
1.system(“cls”);清屏
2.system(“pause”);执行时暂停
3.system(“路径”);通往某个文件或者网址(不过需要完整的域名)
另外需要注意到的是,在通往新程序时,如果当前程序没有必要继续存在,需要关闭,防止占用内存(exit(0);)
Part3 具体功能实现
由简入难,首先完成订单查询部分。
这个部分不需要什么特殊难度,只需要将txt文档中的数据按规格读取,显示在屏幕中即可。
流程可以简化为:打开文件流,按格式读取txt文件的内容,格式化输出显示屏幕,这里我另外写了一个back函数,内容和菜单部分大部分相同。
因为当初为了简单些,所以基本每个功能都是分开的,但是这样就造成了不好调取另一个文件内的函数,于是有几种办法可以在两个独立文件之间调取函数:
1.使用#include。实际代码是
#include“文件路径名”
这个就可以将另一个文件里的所有函数都包括了。当然,main也会被包括,如果没注意到这点的话,可能会被警告previous main() function has been included之类的。
2.这个还没试,就是通过函数地址来实现。如一般数据类型在被声明之后,都会有一个固定的地址。你可以使用指针,或直接访问地址来获取数据。当然,函数也有地址,也可以使用类似方式访问。实际代码会在过后几天写在下面
然后可以进行较复杂功能的实现。比如按上座率进行历史航班查询。
需要解决的重点我以粗体显示。也就是如何排序和在一大堆航班那中只显示历史航班的问题。
上座率是根据已给数据计算的,这里给一个简单的一行数据提取。
CA1351 中国国际航空公司 08:00 11:00 320 2080 1380 8 150 09/07/16 0
航班号 航班公司 起飞点 降落点 机型 票价1 票价2 余票1 余票2 起飞日期 已购
计算公式-->上座率=已购买票数/(已购买票数+余票数)
接下来是进行排序的部分,一般大量排序使用的方法是冒泡排序。这个原理很简单,就是两个两个比较,if (前一个如果大于后一个),{.....}(那么交换)。在进行一遍交换后,继续交换,直到不能交换为止。另:结构体之间可以进行直接交换,如下图。
这是代码部分
如果看得不清楚,可以点击这里。事实上,在百度百科,甚至其他博客给的方法大同小异,设置一个最大比较数,这样可以绝对排序完毕。这是一种方式,也可以写成别的方式。下面是基本思路。
比如在设置声明一个多余的N。在每次比较后,如果出现交换情况,就加1,每次大循环初始化一次。如果在每次大循环结束时,N!=0,进行下一次循环,如果N==0,就说明冒泡结束。
根据上座率排序完成后,另一个需要解决的是时间问题。事实上C语言有直接获取当前系统时间的代码,这里直接给出。
如果想获取该段代码,点击这里。
那么,如何利用这段代码来筛选出历史航班呢。简单说来,就是当前时间和txt文件的时间进行第比
CA1351 中国国际航空公司 08:00 11:00 320 2080 1380 8 150 09/07/16 0
航班号 航班公司 起飞点 降落点 机型 票价1 票价2 余票1 余票2 起飞日期 已购
再次展示出来,就不需要翻页了。需要进行对比的就是起飞时间。但实际上,起飞时间从txt文档里赋值到结构体时,其数据类型是char,是字符串。所以不能和time函数获取的时间进行直接对比。解决办法是:
设置一个新的字符串datex,然后读取起飞时间的字符串(假设声明为buy.date,内容就是起飞日期,为09/07/16)。在c语言中,往往可以对字符串进行数组处理,先读取buy.date[1],就是9,然后赋值到datex[0],继续将buy.date[3]&&buy.date[4]赋值到datex[1]&&datex[2]中,这样得到一个datex="907",使用atoi函数进行字符串转化为int类型,得到一个整型数据907,然后将获取到的整形数据和time函数得到的当前时间进行对比,就可以获得跟历史航班对比的条件。
一些识别身份之类的细节,等明天继续。