学号2012-2013学年第二学期1208010217《高级语言程序设计》课程设计报告题目:学生考勤管理系统专业:计算机科学与技术班级:12(2)班姓名:李天栋指导教师:孙淮宁成绩:计算机与信息工程系2013年6月30日目录
1设计内容及要求.........................................31.1设计内容.....................................................31.2设计任务及具体要求...........................................32概要设计.........................................................42.1该系统的功能简介.............................................42.2总体程序框图.................................................42.3各个模块之间的主要关系........................................53设计过程或程序代码............................................53.1各个模块的程序流程图及运行界面................................53.2对关键代码加以分析说明.......................................104程序调试分析.....................................................145小结.............................................................17致谢...........................................................17参考文献...........................................................17附:源程序..........................................................181设计内容及要求
1.1设计内容学生考勤信息记录了学生的缺课情况,它包括:缺课日期、第几节课(连续用begin-end的形式表示)、课程名称(课程名称中不会出现空格)、学生姓名、缺课类型(迟到、早退、请假及旷课)。1.2设计任务及具体要求任务:(1)给出软件结构,说明各模块的功能。(2)设计相关的类,并说明该类的作用。特别要用图形说明类之间的继承关系。(3)编写代码具体要求:(1)定义相关的数据,比如:学生学号学生姓名课程名称上课时间到课时间出勤状态[旷课/病假/事假/迟到/早退](2)能够进行以下操作:A.能够对数据进行增、删、改、查操作。B.能够按学号顺序列出某一课程的出勤状况2概要设计
2.1系统的功能简介考勤管理系统,有以下功能: (1).录入学生的缺课记录:从键盘输入数据(提示:为避免重复从键盘输入数据,测试时可将数据存储在文件中,利用输入重定向功能读入),输入格式为:缺课日期 第几节课 课程名称学生姓名 缺课类型 每行一条纪录。 例如: 2013-04-29 2-4 中国近现代文学史 李云龙 迟到 2013-04-28 2-4 大学生社交礼仪 马志鹏 旷课 (2).修改某个学生的缺课记录:可以对缺课纪录的任意部分进行修改,然后显示一下修改后的纪录。 (3).查询某个学生的缺课情况:查询结果按照日期升序排序,同一天内按照所缺课程的时间升序排序。 (4).统计某段时间内(以天为单位),旷课学生姓名及旷课节数,查询结果先按旷课节数降序排序,旷课节数相同的学生按姓名升序排序; (5).统计某段时间内,有学生旷课的课程及旷课人次,按旷课人次由多到少排序,旷课人次相同的课程按课程名称升序排序。 2.2总体程序框图
修改某学生的缺课信息查询某学生的缺课信息统计某段时间内旷课学生姓名及旷课节数统计某段时间内,有学生旷课的课程及旷课人次录入学生的缺课信息学生考勤管理系统录入缺课学生的姓名、缺课日期、缺课节次、缺课课程名称、缺课类型修改某学生的缺课日期、缺课节次、缺课课程名称、缺课类型查询某学生的缺课日期、缺课节次、缺课课程名称、缺课类型按时间升序显示旷课学生姓名及旷课节数按时间升序显示有学生旷课的课程及旷课人次2.3各个模块之间的主要关系本程序用结构体新定义一种数据类型,系统包含一个主函数和6个子函数(insert(),check(),rivise(),Delete(),output(),deleteallchain()),6个子函数分别用来插入、查询、修改、删除、输出、收回动态分配的空间,在主函数中通过调用子函数来实现所需功能。3设计过程或程序代码3.1各个模块的程序流程图及运行界面学生考勤管理系统中四个类的类层次图为:
Time类Xinxi类qstu类qkechen类图2学生考勤管理系统中中四个类的类层次图学生考勤管理系统中各功能模块的实现:菜单统计某段时间内,有学生旷课的课程及旷课人次统计某段时间内旷课学生姓名及旷课节数查询某学生的缺课信息录入学生的缺课信息修改某学生的缺课信息图3学生考勤管理系统中菜单函数的功能图1、学生缺课信息录用功能模块:
学生缺课信息录用功能录入学生缺课节次录入学生缺课日期录入学生缺课课程名称录入缺课学生姓名录入学生缺课类型图4学生缺课信息录入的功能1、修改某学生缺课信息功能的模块图修改某学生缺课信息功能修改某学生的缺课节次修改某学生的缺课日期修改某学生的缺课课程修改某学生的缺课类型图5学生考勤管理系统修改学生缺课信息功能图3查询某学生缺课信息功能图:查询某学生缺课信息功能输入要查询的学生姓名
显示要查询的学生缺课信息继续查询或返回菜单图6学生考勤管理系统查询学生缺课信息功能图4、统计某段时间内旷课学生姓名及旷课节数功能模块:统计某段时间内旷课学生姓名及旷课节数功能统计某段时间内旷课学生姓名及旷课节数显示某段时间内旷课学生姓名及旷课节数返回菜单图7统计某段时间内旷课学生姓名及旷课节数功能图5、统计某段时间内,有学生旷课的课程及旷课人次功能模块统计某段时间内,有学生旷课的课程及旷课人次统计某段时间内,有学生旷课的课程及旷课人次功能显示某段时间内,有学生旷课的课程及旷课人次返回菜单图8统计某段时间内,有学生旷课的课程及旷课人次功能图
Tine-year:int-month:int-day:int+time()+~time()Xinxi-name[20]:char-kechen[10]:char-jieci[3]:char-queke[10]:char-tm[3]:char-tiaoshu:int=0+xinxi()+~xinxi()+setname(n:char*):void+settime(y:int,m:int,d:int):void+setqklx(qk:char*):void+setjieci(jc:char*):void+setkechen(kc:char*):void+getname():char*+getqklx():char*+getjieci():char*+getkechen():char*+gettime():int*qkechen-renci:int-kcshu:int=0+qkechen()+~qkechen()+getrc(st1[]:xinxi,n:int,j:int):int+qkn[20]:charqstu-jieshu:int-qst:int+qstu()+~qstu()+print(st1:xinxi&):void+getjs(st1[]:xinxi,n:int,j:int):int+setjs(js:int=0):int+qstn[20]:char基类派生类派生类派生类图9学生考勤管理系统中四个类的UML图
3.2对关键代码加以分析说明#include
#include#includestructchuqin{//用结构体定义一个新的数据类型出勤chuqincharname[10],number[20],subject[15],time1[15],time2[15],zhuangtai[10];chuqin*next;};//------------------------------------------------------------------------------------voidinsert(chuqin*&head1)//创建单向链表并插入数据{chuqin*n;//定义一个chuqin类型指针nwhile(1)//一个死循环{n=newchuqin;//用new运算符动态分配一个chuqin空间给指针ncout<<"请依次输入姓名、学号、课程、上课时间、到课时间、出勤状态(第一个字母输入N表示结束插入):n";//输出提示信息cin>>n->name;//输入姓名if(n->name[0]=="N")//判断刚才输入的第一个字母是否为N{deleten;//若输入的第一个字母为N,则用delete运算符收回刚才动态分配的空间break;//跳出死循环,结束插入}cin>>n->number>>n->subject>>n->time1>>n->time2>>n->zhuangtai;//若条件不满足,则继续输入学号、课程、上课时间到课时间等if(!head1)head1=n,n->next=NULL;/*假如head指针为空,则直接让head指针指向n指向的存储空间(head=n),将n->next赋值为空,即不指向任何存储空间*/elsen->next=head1,head1=n;//若head指针非空,则将n指向的存储空间插在最前面}}//------------------------------------------------------------------------------------voidcheck(chuqin*head)//子函数实现查询功能{charnumber1[10];//定义一个字符串数组name1临时存放要查询的姓名intflag=1;cout<<"请输入要查询的学号:";//输出提示信息cin>>number1;//输入要查询的学号while(head)//用while循环检索查询是否有匹配的姓名{if(strcmp(number1,head->number)==0)//用函数strcmp()判断是否有匹配姓名{if(flag)
{cout<<"找到!n"<name<number<subject;cout<time1<time2<zhuangtai<next;//更新head指针,是它指向下一个chuqin类型的存储空间}if(flag)cout<<"未找到!";//假如flag值为1,则输出"未找到!"}//------------------------------------------------------------------------------------voidrivise(chuqin*head)//定义子函数实现修改功能{charname1[10];//定义一个字符串数组name1临时存放要修改的姓名cout<<"请输入修改的姓名:";//输出提示信息cin>>name1;//输入要查询的姓名while(head){if(strcmp(name1,head->name)==0){cout<<"找到此人!t"<<"请输入此人新的信息(依次为:课程、上课时间、到课时间、出勤状态)n";cin>>head->subject>>head->time1>>head->time2>>head->zhuangtai;//输入新的课程、上课时间、到课时间break;}head=head->next;}if(head==NULL)cout<<"未找到此人!无法修改!n";}//------------------------------------------------------------------------------------voidDelete(chuqin*&head)//定义子函数实现删除功能{charname1[10];intflag=1;chuqin*p1=NULL,*p2,*p3=head;cout<<"请输入要删除的姓名:";cin>>name1;while(head){if(strcmp(name1,head->name)==0){cout<<"找到此人!t将此人的全部信息删除n";if(p1==NULL){p1=head;
head=head->next;deletep1;}elseif(p2->next==NULL){p1->next=NULL;deletep2;head=p3;}else{p1->next=p2->next;deletep2;head=p3;}flag=0;break;}p1=head;head=head->next;p2=head;}if(flag){cout<<"未找到此人!无法删除!n";}}//------------------------------------------------------------------------------------voidoutput(chuqin*head)//定义子函数实现输出功能{if(head==NULL)cout<<"记录为空!";else{cout<<"所有的记录依次为:n"<name<number<subject;cout<time1<time2<zhuangtai<next;}}//-----------------------------------------------------------------------------------------------voiddeleteallchain(chuqin*head)//定义子函数来收回所有动态分配的存储空间{
chuqin*p;while(head){p=head;head=head->next;deletep;}}//------------------------------------------------------------------------------------voidmain()//主函数{inta;//定义一个整型变量chuqin*head;//定义一个head指针head=NULL;while(1)//死循环{cout<<"请输入一个数字(1表示插入,2表示查询,3表示修改,";cout<<"4表示删除,5表示输出,6表示跳出整个程序)n";//输出提示信息cin>>a;//输入一个整型值if(a==6)break;//若该值为6则跳出整个循环switch(a)//若a值不为6,执行一下程序{case1:insert(head);break;case2:check(head);break;case3:rivise(head);break;case4:Delete(head);break;case5:output(head);break;default:cout<<"输入有误,请重新输入n";/*若输入的a值不符合要求,则输出提示信息*/}}deleteallchain(head);//调用函数deleteallchain()收回所有动态分配的存储空间}4设计结果与分析
系统调试程序编写完成后,我进行了程序调试。调试过程中,出现了以下问题:当录入学生缺课信息时,出现了无法退出录入函数的情况,主要原因是输入函数中while循环语句没有设置跳出循环的条件。除此之外,修改某学生缺课信息时,如果没有该学生的缺课信息,系统没有提示要用户没有该学生信息,也没有建议用户重新输入要修改的学生姓名。由此我发现了该系统修改学生缺课信息的不足。于是,我对修改函数进行了修改与完善。在该函数中,我设置了if语句用来判断系统中是否存有用户要修改的学生缺课信息。如果没有,则提示用户没有该学生的缺课信息,用户可以选择继续输入要修改的学生姓名或者退出修改环节,可以在 循环语句中改变循环进行的条件从而终止循环。至于在修改后显示该学生的缺课信息,运行时发现系统并没有做到这一点,对源代码检查之后发现是修改函数中显示函数的调用出了点问题,显示函数的实参有误,应该把要修改缺课信息的学生所在的信息类对象作为实参,然后用显示函数的形参引用实参对象,通过对象调用信息类的成员函数,从而实现显示功能。
5小结
这次的课设,我写的程序也没有什么独到之处,函数参数设定不够完美,整个程序中对指针的运用比较少。对于指针的巧妙运用把握得不够。还有就是容易missing;或者是},这都是些应该改掉的毛病。除此以外,我对于怎样用文件保存数据还没深入了解,但我将数据保存到文件后又因文件打不开而无法访问文件中的数据,对于动态链表,我不是很清晰,也就是说无法做到熟稔运用。在调试过程中我发现在主函数中把字符数组名name赋值给字符指针p即p=name;然后用cout<
#include#includeusingnamespacestd;//时间类classtime{public:time(inty=0,intm=0,intd=0){year=y;month=m;day=d;}~time(){}protected:intyear,month,day;};//信息类classxinxi:publictime{public:xinxi(){tiaoshu++;}~xinxi(){tiaoshu--;}voidsetname(char*n){strcpy(name,n);}voidsettime(inty,intm,intd){year=y;month=m;day=d;}voidsetqklx(char*qk){strcpy(queke,qk);}voidsetjieci(char*jc){strcpy(jieci,jc);}voidsetkechen(char*kc){strcpy(kechen,kc);}char*getname(){char*n;n=name;returnn;}char*getqklx(){char*qk=queke;returnqk;}char*getjieci(){char*jc;jc=jieci;returnjc;}char*getkechen(){char*gk;gk=kechen;returngk;}int*gettime();private:charname[20];//姓名charjieci[3],kechen[10],queke[10];inttm[3];staticinttiaoshu;};intxinxi::tiaoshu=0;int*xinxi::gettime(){int*t;t=tm;tm[0]=year;tm[1]=month;tm[2]=day;returnt;
}//缺课课程类classqkechen:publicxinxi{public:qkechen(){kcshu++;}~qkechen(){kcshu--;}intgetrc(xinxist1[],intn,intj);charqkn[20];//公有数据成员private:intrenci;staticintkcshu;//静态数据成员};intqkechen::kcshu=0;intqkechen::getrc(xinxist1[],intn,intj){renci=0;renci++;for(inti=0;i