MFC学生信息管理系统(共32页).doc
精选优质文档-倾情为你奉上重庆交通大学信息科学与工程学院报告书专 业:计算机科学与技术课程 名称:信息技术实训I题 目:学生信息管理系统班 级:2010级,计科四班设 计 者: 韩旭辉完 成 时 间:2011年 11月1 日至2011年 11月27 日重庆交通大学信息科学与工程学院任务书课 程信息技术实训班级2010计科专业信息技术方向指导教师徐毅xuyitiger题 目自选完成时间2011年10月19日至2011年11月 2日主要内容要求:完成一个完整功能实现的MFC程序开发,报告需包括以下内容1系统分析与设计(总体设计和模块详细设计)。2系统编程与实现(必须有源码和注释)。3功能测试计划内容和结果。4存在问题与不足及对策。设计报告要求1封面:(格式见前页)2课程设计任务书3课程设计报告: 功能需求分析 系统设计功能模块详细设计和流程图 设计代码 心得体会和参考资料说明:第14周周三下午34节一班提交打印设计报告及软件,回答老师提问,周四下午34节二班提交打印设计报告及软件,回答老师提问。报告文字不少于5000 字,并装订成册。考核内容与方式1 平时成绩: 20%2 设计报告: 40%3 程序与演示,回答问题:40%版面要求1 题目用黑体三号,段后距18磅(或1行),居中对齐;2 标题用黑体四号,段前、段后距6磅(或0.3行);3 正文用小四号宋体,行距为1.25倍行距;4 标题按“一”、“”、“1”、“”顺序编号。上机时间安排星期周次一二三四五六日1819 课程设计报告一. 功能需求分析:(一). 实现简单地使用用户名和密码来进入登陆界面,以加强DOS运行窗口的可视化效果。如果输入的用户名和密码有一项为空或者是输入的用户名和密码是错误的,则不能进入系统。(二). 实现任意添加学生的信息,包括添加学生的个人学号,姓名,性别,专业,年级出生日期。输入不允许为空,并且有提示功能,如果输入的时候有一项为空则提示输入相关的信息。(三). 实现删除学生的信息,包括删除学生的个人学号,姓名,性别,专业,年出生日期。根据编号进行用户权限检查,在数据库中进行查找相符和的记录,如果编号在数据库中不存在相应记录,则提示不能删除,否则可以删除该记录。输入不允许为空,并且有提示功能,如果输入的时候为空则提示输入相关的信息。(四). 实现编辑学生的信息,包括编辑学生的个人学号,姓名,性别,专业,年级,出生日期。输入不允许为空,并且有提示功能,如果输入的时候有一项为空则提示输入相关的信息。(五). 实现查询学生的信息,包括编辑查询学生的个学号,姓名,性别,专业,年级,出生日期。输入不允许为空,并且有提示功能,如果输入的时候为空则提示输入相关的信息。(六). 实现保存学生的信息,包括保存学生的个人学号,姓名,性别,专业,年级,出生日期。二系统分析:(一). 该实验在Visual C+ 的MFC 中完成,通过可视化编程实现实验的要求。用工程MFC AppWizard(exe) 建一个工程文件Stuinfor。(二).该系统的MFC的实验共分为四个类:Add 实现学生信息的添加Delete 实现学生信息的删除Modify 实现学生信息的修改Search 实现学生信息的查找(三).由系统的功能分析,可把系统分成若干个对话框窗口的形式,分别实现系统的输入,修改,删除,查询功能。每一个对话框定义一个类分别用来完成各个功能的实现。整个系统的功能在菜单内完成,分别是在学生信息管理下的学生信息录入,学生信息的删除,学生信息的修改,学生信息的查询。可视化的窗口五个,分别实现用户的登录,学生信息的录入,修改,删除,查找。通过窗口的各个按键对学生信息进行操作。(四).由于整个系统的功能比较简单,所以实现所需的控件和代码不会太多,各个控件的变量以及控件的点击映射函数都是比较简洁的,整体来说,该系统是一个比较简洁的系统,他通过简明易懂的方式实现了实验的目的,对初步接触MFC的学生来说是一个非常好的实验教程。三功能模块详细设计和流程图(一).功能模块详细设计1首先完成系统的登录窗口的设计,按ctrl+r键,弹出插入资源对话框,选择dialog新建,在该对话框的属性里将标题改为登录。然后在新建的窗口中分别加入图中的各个控件,其中包括两个静态文本,两个编辑框。其中静态文本分别定为用户和密码,其相对应的编辑框的ID分别按默认的IDC_EDIT1,IDC_EDIT2不需要更改。在窗口中的确认键的属性里将标题改为登录,取消不变,其ID分别为IDOK,IDCANCEL不变。调整各个控件的位置,如图。2完成学生信息的输入窗口,将窗口标题改为输入。同理,创建一个对话框,加入静态文本,学生信息录入,学号,姓名,性别,专业,年级,出生日期,ID取默认的,其中与学号,姓名,性别(男,女),专业,年级,出生日期,录入,取消相对应的编辑框或是按钮的ID分别是IDC_EDIT2,IDC_EDIT3,IDC_RADIO1,IDC_RADIO2,IDC_EDIT4,IDC_EDIT5,IDC_EDIT6,IDOK,IDCANCEL如图所示加入个控件并设置相应的ID,调整个控件的位置。3同上,完成学生信息管理系统的删除窗口,将窗口属性里的标题改为删除,加入静态文本,编辑框,按钮,组框,如下图所示,其中静态文本的ID取其默认值,将确认键的标题改为删除,其余控件的ID也保持默认值不变。调整控件位置如图。组框的属性里的标题改为查询结果,用于显示查寻结果的,为下面信息的删除做好铺垫。如果没有查到要找的信息,系统也会给出提示,同时,查询结果里没有任何信息。同时为查询按钮映射一个成员函数OnButton1(),用于完成查询按钮的功能的实现。4查询窗口的设计也是一样,在窗口属性里将标题改为查询,如下图加入各个静态文本和控件,同上,各个控件的ID取其默认值,删除取消键,将确认键属性里的标题改为查询,用于执行数据的查询,同上,此对话框里也有一个查询结果的组框用于信息的输出,按键ctrl+w键,在MFC ClassWizard里为查询键加入函数映射OnOK(),此函数的功能是查询记录中的数据。5创建一个对话框如图,图中加入的控件有静态文本,编辑框,旋转按钮,单选按钮,列表控件,删除原图中的确认,取消键,ID取其默认值。6在ResourceView的Menu的工具栏里添加按键如下,按钮学生信息管理属性里的选择弹出,在学生信息管理下分别加入按键学生信息录入,学生信息的删除,学生信息的修改,学生信息的查询,ID分别为ID_MAdd,ID_MDelete,ID_MModify,ID_MModify 。7在MFC ClassWizard的Member Variables 中分别为类Search,Delete,Modify,Password控件添加变量如下图8在主窗口的类CMainFrame为成员函数添加点击映射,如下图所示,代码在后面。(二) 流程图登录,输入用户名,密码 进入系统 进入系统页面密码输入错误学生信息管理学生信息查询学生信息修改 学生信息删除学生信息录入输入要查询的学号学生基本信息浏览查找选中输入学号查找录入完毕有无有填入修改后的数据修改输出查询结果无恶该学生信息不存在确定要删除该学生的所有信息吗该学生信息不存在确认确定要修改该学生的信息确认该学生信息已删除修改完毕四设计代码(一)录入学生信息void Add:OnOK() UpdateData();if(m_sNum = "" | m_sName = "" | m_sMajor = "" | m_sGrade = "" |m_sBirthday = "" | (!m_cMale.GetCheck() && !m_cFemale.GetCheck() MessageBox("学生信息不完整,无法录入!", "错误");return;/ 查询学号是否存在int flag = 0;CStudent *stu = new CStudent;stu->m_strID = m_sNum;string temp;ifstream in("record.txt");while(in >> temp) if(temp = stu->m_strID) flag = 1;MessageBox("该学号已存在!", "注意");in.close();m_sNum = ""UpdateData(false);/ 只将学号清空,其余信息保留return;in.close();/ 说明该学生信息还没存档,以下将其信息存档write(stu);delete stu;MessageBox("录入完毕!", "成功");clean();/ 清空表格/ TODO: Add extra validation here/CDialog:OnOK();void Add:clean()m_sNum = ""m_sName = ""m_sMajor = ""m_sGrade = ""m_sBirthday = ""m_cMale.SetCheck(0);m_cFemale.SetCheck(0);UpdateData(false);void Add:write(CStudent* stu)stu->m_strName = m_sName;stu->m_strMajor = m_sMajor;stu->m_strGrade = m_sGrade;stu->m_strDate = m_sBirthday;string temp;if(m_cMale.GetCheck()temp = "男"elsetemp = "女"ofstream ou;ou.open("record.txt", ios:app);/ 选择附加模式ou << left;ou << setw(20) << stu->m_strID;ou << setw(20) << stu->m_strName;if(m_cMale.GetCheck()temp = "男"elsetemp = "女"ou << setw(20) << temp;ou << setw(20) << stu->m_strMajor;ou << setw(20) << stu->m_strGrade;ou << setw(20) << stu->m_strDate;ou << endl;ou.close();(二)查询信息代码void Delete:OnOK() if(!searched) MessageBox("请先查询!", "注意");return;if(!find) MessageBox("该学生信息不存在,无法删除!", "错误");return;int mbr = MessageBox("确定要删除该学生的所有信息吗?", "注意", MB_YESNO|MB_ICONQUESTION);if(mbr = IDNO) return;int k;ifstream in("record.txt");for(int i = 1; i <= index; i+) string a;in >> a;k = a.size();/为了得到文件指针的偏移量if(i = index)/ 说明文件指针到达要删除那行了,但偏移了k字节break;in.ignore(1024, 'n');in.seekg(-k, ios:cur);/ 让指针跳到要删除那行的行首int offset = in.tellg();/ 记下指针位置,便于写新的记录fstream ou;/ 这是文件流,不是输出流!ou.open("record.txt");ou.seekp(offset, ios:beg);/跳到删除行string str;getline(in, str);int size = str.size();/ 得到删除行的大小for(i = 0; i < size; i+)ou << " "/ 用空格代替删除行MessageBox("该学生信息已删除!", "成功");m_sNum = ""clean();searched = false;/ 在表格是空的情况下要求先查询/ TODO: Add extra validation here/CDialog:OnOK();void Delete:OnButton1() find = false;/ 该变量表示是否找到searched = true; / 该变量表示是否进行过查询UpdateData();clean();/ 清空上次查询结果,仅保留现在输入的学号if(m_sNum = "") MessageBox("学号为空,无法查询!", "错误");searched = false;return;index = 0;/ 表示要查找的学生信息在哪一行,作为删除的索引ifstream in("record.txt");string temp, str;while(in >> temp) index+;str = m_sNum;if(str = temp) find = true;/ 找到了in >> temp;m_sName = temp.c_str();in >> temp;if(temp = "男")m_cMale.SetCheck(1);elsem_cFemale.SetCheck(1);in >> temp;m_sMajor = temp.c_str();in >> temp;m_sGrade = temp.c_str();in >> temp;m_sBirthday = temp.c_str();UpdateData(false);in.close();break;in.ignore(1024, 'n');in.close();if(!find) MessageBox("该学生信息不存在!", "注意");m_sNum = ""UpdateData(false);searched = false;/ TODO: Add your control notification handler code herevoid Delete:clean() m_sMajor = ""m_sGrade = ""m_sBirthday = ""m_cMale.SetCheck(0);m_cFemale.SetCheck(0);UpdateData(false);(三)修改学生信息代码void Modify:OnOK() / TODO: Add extra validation hereUpdateData();ifstream in("record.txt");int r = 0, k;string a;while(r != m_nRecord) in >> a;k = a.size();r+;if(r = m_nRecord)break;in.ignore(1024, 'n');in.seekg(-k, ios:cur);int offset = in.tellg();string temp, str;int flag = 0;/ 用来判断信息是否改动for(int col = 0; col < 6; col+) in >> temp;switch(col) case 0:if(m_sNum != temp.c_str() flag = 1; break;case 1:if(m_sName != temp.c_str()flag = 1;break;case 2:if(m_cMale.GetCheck()str = "男"elsestr = "女"if(temp != str)flag = 1;break;case 3:if(m_sMajor != temp.c_str()flag = 1;break;case 4:if(m_sGrade != temp.c_str()flag = 1;break;case 5:if(m_sBirthday != temp.c_str()flag = 1;break;if(flag)break;if(flag = 0) MessageBox("该学生信息没有改动!", "注意");return;if(m_nRecord = 0 | index = 0) MessageBox("没有学生信息!", "错误");return;/ 查看新改的学号是否已存在ifstream infile("record.txt");int rd = 0;/ 因为要排除原来那行,是与其它行比较看是否重号while(infile >> temp) rd+;if(temp.c_str() = m_sNum && rd != m_nRecord) MessageBox("该学号已存在, 请重新输入!", "注意", MB_ICONWARNING);show(m_nRecord);return;infile.ignore(1024, 'n');infile.close();int mbr = MessageBox("确定要修改该学生的信息吗", "注意", MB_YESNO|MB_ICONQUESTION);if(mbr = IDNO) show(m_nRecord);/ 显示原来信息return;CStudent *stu = new CStudent;stu->m_strID = m_sNum;stu->m_strName = m_sName;if(m_cMale.GetCheck()stu->m_strSex = "男"elsestu->m_strSex = "女"stu->m_strMajor = m_sMajor;stu->m_strGrade = m_sGrade;stu->m_strDate = m_sBirthday;/ 将信息存入变量,便于比较stu->savetofile(offset);delete stu;LVITEM lvi;lvi.mask = LVIF_IMAGE | LVIF_TEXT;lvi.iItem = m_nRecord;in.seekg(offset, ios:beg);int c = 0;while(in >> str) lvi.iSubItem = c;lvi.pszText = (LPTSTR)(str.c_str();if(c = 0)m_cList.InsertItem(&lvi);elsem_cList.SetItem(&lvi);c+;if(c = 6) break;m_cList.DeleteItem(m_nRecord-1);MessageBox("修改完毕!", "成功");/CDialog:OnOK();void Modify:show(int r)clean();if(r <= 0)return;if(r > index)return;ifstream in("record.txt");int k = 0;for(int i = 1; i <= r; i+) string a;in >> a;k = a.size();if(i = r)break;in.ignore(1024, 'n');/同样,也是为了得到要显示那行in.seekg(-k, ios:cur);/退回到行头m_nRecord = r;string temp;for(int col = 0; col != 6; col+) in >> temp;switch(col) case 0:m_sNum = temp.c_str();break;case 1:m_sName = temp.c_str();break;case 2:if(temp = "男")m_cMale.SetCheck(1);elsem_cFemale.SetCheck(1);break;case 3:m_sMajor = temp.c_str();break;case 4:m_sGrade = temp.c_str();break;case 5:m_sBirthday = temp.c_str();break;UpdateData(false);in.close();void Modify:clean()m_nRecord = 0;m_sNum = ""m_sName = ""m_sGrade = ""m_sMajor = ""m_sBirthday = ""m_cMale.SetCheck(0);m_cFemale.SetCheck(0);UpdateData(false);BOOL Modify:OnInitDialog() CDialog:OnInitDialog();/ TODO: Add extra initialization hereindex = 0;/ 行数ifstream infile("record.txt");string str;while(infile >> str) / 为了得到总共的记录数(行数)index+;infile.ignore(1024, 'n');infile.close();m_cList.InsertColumn(0, "学号", LVCFMT_LEFT, 130);m_cList.InsertColumn(1, "姓名", LVCFMT_LEFT, 100);m_cList.InsertColumn(2, "性别", LVCFMT_LEFT, 70);m_cList.InsertColumn(3, "专业", LVCFMT_LEFT, 120);m_cList.InsertColumn(4, "年级", LVCFMT_LEFT, 80);m_cList.InsertColumn(5, "出生日期", LVCFMT_LEFT, 100);ifstream in("record.txt");LVITEM lvi;for(int r = 0; r < index; r+) lvi.mask = LVIF_TEXT;lvi.iItem = r;int c = 0;while(in >> str) lvi.iSubItem = c;lvi.pszText = (LPTSTR)(str.c_str();if(c = 0)m_cList.InsertItem(&lvi);elsem_cList.SetItem(&lvi);c+;if(c = 6)break;in.close();DWORD dwStyleEx;/ 网格风格dwStyleEx = LVS_EX_FULLROWSELECT | LVS_REPORT | LVS_EX_ONECLICKACTIVATE; m_cList.SetExtendedStyle(dwStyleEx);m_cSpin.SetRange(0, index);return TRUE; / return TRUE unless you set the focus to a control / EXCEPTION: OCX Property Pages should return FALSEvoid Modify:OnDeltaposSpin2(NMHDR* pNMHDR, LRESULT* pResult) NM_UPDOWN* pNMUpDown = (NM_UPDOWN*)pNMHDR;/ TODO: Add your control notification handler code hereint r = m_cSpin.GetPos();if(pNMUpDown->iDelta = 1)/ 说明点击了Spin的往上箭头show(r+1);else if(pNMUpDown->iDelta = -1)/ 说明点击了Spin的往下箭头show(r-1);*pResult = 0;void Modify:OnClickList3(NMHDR* pNMHDR, LRESULT* pResult) / TODO: Add your control notification handler code herePOSITION pos = m_cList.GetFirstSelectedItemPosition();int r = m_cList.GetNextSelectedItem(pos);show(r+1);*pResult = 0;void Modify:OnKeydownList3(NMHDR* pNMHDR, LRESULT* pResult) LV_KEYDOWN* pLVKeyDow = (LV_KEYDOWN*)pNMHDR;/ TODO: Add your control notification handler code herePOSITION pos = m_cList.GetFirstSelectedItemPosition();int r = m_cList.GetNextSelectedItem(pos);if(GetKeyState(38) < 0)/ upshow(r);if(GetKeyState(40) < 0)/ downshow(r+2);*pResult = 0;(四)查询信息代码void Search:OnOK() / TODO: Add extra validation hereUpdateData();m_sName = ""m_sMajor = ""m_sGrade = ""m_sBirthday = ""m_cMale.SetCheck(0);m_cFemale.SetCheck(0);UpdateData(false);/ 清空上次查找记录,只保留现在输入的学号if(m_sNum = "") MessageBox("学号为空,无法查询!", "错误");return;ifstream in("record.txt");string temp, str;int flag = 0;while(in >> temp) str = m_sNum;if(temp = str) flag = 1;/ 说明找到了in >> temp;m_sName = temp.c_str();/ 类型转换in >> temp;if(temp = "男")m_cMale.SetCheck(1);elsem_cFemale.SetCheck(1);in >> temp;m_sMajor = temp.c_str();in >> temp;m_sGrade = temp.c_str();in >> temp;m_sBirthday = temp.c_str();UpdateData(false);in.close();break;in.ignore(1024, 'n');in.close();if(!flag) MessageBox("该学生信息不存在!", "注意");m_sNum = ""UpdateData(false);/CDialog:OnOK();(五)登录信息代码BOOL CStuinforApp:InitInstance()AfxEnableControlContainer();int nCount=0;while(nCount<3)Password passdlg;if(passdlg.DoModal()=IDOK)if(strcmp(passdlg.m_strusername,"hanxvhui")!=0)|(strcmp(passdlg.m_strpassword,"0524")!=0)MessageBox(NULL,"用户名或口令错误,请重试","错误信息",MB_OK|MB_ICONERROR);nCount+;elsebreak;elsereturn FALSE;if(nCount>=3)MessageBox(NULL,"口令输入已超过三次,请退出!","错误信息",MB_OK|MB_ICONERROR);return FALSE;(六)写入学生基本信息write:write()CStudent *stu; stu->m_strName = m_sName;stu->m_strMajor = m_sMajor;stu->m_strGrade = m_sGrade;stu->m_strDate = m_sBirthday;string temp;if(m_cMale.GetCheck()temp = "男"elsetemp = "女"ofstream ou;ou.open("record.txt", ios:app);/ 选择附加模式ou << left;ou << setw(20) << stu->m_strID;ou << setw(20) << stu->m_strName;if(m_cMale.GetCheck()temp = "男"elsetemp = "女"ou << setw(20) << temp;ou << setw(20) << stu->m_strMajor;ou