贺老师教学链接 C语言及程序设计提高 本课讲解 参考银行系统的上一个版本
开发指导
该银行储蓄系统的目标,将是设计一个可以支持多名储户开户、存款、取款、转帐、改密、挂失、解挂、销户等功能的系统,以此初步体会完整系统开发,综合运用已经学过的知识。
以下提示中,给出了多个层次的需要,可以先保证最基本、最简单功能的实现,迭代式地完善,实现“改进意见”中的要求。
按照模块化程序设计要求,充分利用函数实现功能。
功能 |
图 示 |
说明和提示 |
||||||||||||||||||||||||||||||
登录 |
输入业务员用户名和密码。3次输入不正确,强制退出程序。 本讲方案 只设一名业务员,用全局变量表示其用户名和密码,写在文件中,程序开始时读入。 改进意见 可以用一个文本文件保存多名业务员的信息,在程序开始执行时,读入业务员信息,以支持多名业务员登录。 用文件保存密码时,一般不用“明文”,而要加密处理。 |
|||||||||||||||||||||||||||||||
系统菜单 |
当业务员信息验证后,进入到系统主控菜单,由业务员为储户办理各种业务,直到按0后退出。 储户的信息都保存在文件中,在处理业务前需要将信息读入,系统退出前需要将数据保存到文件中。 |
|||||||||||||||||||||||||||||||
开户 |
账号由系统自动分配 确认密码和密码不同时要拒绝。 开户后,账号状态为0-正常 |
增加一个银行账户(银行卡) 银行账户信息如下表:
理想方案应该采用结构体,限于我们学习到的技术的限制,我们采用多个数组存储数据 #define upNum 2000 //系统最多容纳的用户数
int user[upNum][3]; //账号、密码、状态 char name[upNum][10]; //用户名 double balance[upNum]; //账户余额 //以上相同行下标,描述同一人的信息 改进意见 1、要求密码是6位数字字符 2、用静态数组太浪费空间,改用动态数组 3、将用户信息改为链表,是更合适的做法 4、保存密码时,可以考虑加密。 5、账户状态state可以定义为枚举型。 |
||||||||||||||||||||||||||||||
销户 |
输入账号,确认后,将余额全部取完,并将状态state置为2-销户。 办理销户的账号,其状态必须为“0-正常”(下同)。 |
|||||||||||||||||||||||||||||||
存款 |
输入账号、自动出现姓名,输入金额后,记录存款后的余额。 |
|||||||||||||||||||||||||||||||
取款 |
输入账号、金额,记录取款后的余额。 取款额不能超过余额。 密码不对拒绝取款。 |
|||||||||||||||||||||||||||||||
查询 |
输入账号,显示账户信息。 |
|||||||||||||||||||||||||||||||
转账 |
输入转出账号、金额、转入账户,记录转账后的余额。 账户必须存在,扣除转出账户的金额,要与转入账户增加的金额相同。 |
|||||||||||||||||||||||||||||||
挂失 |
输入账号,将状态改变为 1-挂失 |
|||||||||||||||||||||||||||||||
解挂 |
输入账号,将状态为 1-挂失 的账户状态改为 0-正常 |
|||||||||||||||||||||||||||||||
改密 |
用新密码替代旧密码。 新密码必须与确认密码相同。 |
|||||||||||||||||||||||||||||||
计算利息 |
(这属于银行后台管理的功能,不体现在界面上,以上的设计尚不能支持真实系统中的计息功能,故将此作为选做功能。) |
现行银行计息的规则是,每半年计算一次利息,并将利息加到账号余额中。 如果要计算利息,需要将每一笔存、取、转的信息记录下来,以获得各个时间段内的金额,完成利息计算。 请自行设计有关的数据结构完成设计 |
下面是源代码:
#include <stdio.h> #include <conio.h> #include <stdlib.h> #include <string.h> #include <ctype.h> #define upNum 2000 //系统最多容纳的用户数 int user[upNum][3]; //账号、密码、状态 char name[upNum][10]; //用户名 double balance[upNum]; //账户余额 //以上相同行下标,描述同一人的信息 int N; //实际的用户数目 int pass(); //业务员登录 void readData(); //开始前从文件中读数据,存在数组中 void writeData(); //程序结束前,将数组中的数据写入到文件中 void work(); //业务驱动 int chooseInMenu(); //显示菜单并由业务员选择 void openAccount(); //开户 void cancelAccount(); //注销账户 void save(); //存款 void withdraw(); //取款 void showAccount(); //查询余额 void transferAccounts(); //转账 void reportLoss(); //挂失 void cancelLoss(); //解除挂失 void updatePassword(); //更改密码 int inputPassword(); //返回键盘输入的密码 int search(int); //根据账号找到用户数据对应的下标 /*主函数:*/ int main() { printf("+----------------------+\n"); printf("+ 欢迎光临CSDN银行 +\n"); printf("+----------------------+\n"); if (pass()) { readData(); work(); writeData(); } return 0; } /* 功能:验证用户密码 返回值:密码正确,返回1; 不能通过密码验证,返回0 */ int pass() { char sNameInFile[20]; //由文件中读出的业务员用户名 char sPassInFile[20]; //文件中保存的密码,这一版本中,用字符保存密码 char sName[20]; //业务员登录时输入的用户名 char sPass[20]; //业务员登录时输入的密码 char ch; int iTry=3; //进入系统时尝试的次数 int right = 0; //要返回的结果:0-不正确 1-正确 FILE *fp; //用于文件操作 //密码保存在文件中,先取出 if ((fp=fopen("worker.txt", "r"))==NULL) { printf("password file cannot open!"); exit(0); } fscanf(fp, "%s %s", sNameInFile, sPassInFile); //从文件中读业务员用户名和密码密码 fclose(fp); //进入系统,密码三次不对将退出 do { printf("请输入业务员用户名:"); scanf("%s", sName); printf("请输入密码:"); int i=0; while((ch=getch())!='\r') //getch在接受输入后,不在屏幕上显示 { sPass[i++]=ch; putchar('*'); //接受任何字符,屏幕上只显示* } sPass[i]='\0'; fflush(stdin); printf("\n"); if(strcmp(sPass,sPassInFile)==0&&strcmp(sName,sNameInFile)==0) { right = 1; break; } else { iTry--; if(iTry>0) printf("超过3次将退出,你还可以尝试%d次!\n", iTry); else { printf("对不起,你不能进入系统\n"); } } } while(iTry); return right; } /* 关于getch()的一点说明: 所在头文件:conio.h 函数用途:从控制台读取一个字符,但不显示在屏幕上 函数原型:int getch(void) 返回值:读取的字符 在不同平台,输入回车,getch()将返回不同数值,而getchar()统一返回10(即\n) 1)windows平台下ENTER键会产生两个转义字符 \r\n,因此getch返回13(\r)。 2)unix、 linux系统中ENTER键只产生 \n ,因此getch返回10(\n)。 3)MAC OS中ENTER键将产生 \r ,因此getch返回13(\r)。 为避免键盘缓存区中未读出的字符影响程序,用fflush(stdin);清除输入缓存区 */ /* 功能:办理业务 */ void work() { int iChoice; //用于选择系统功能 //办理业务 do { iChoice = chooseInMenu(); //从菜单中获得功能代码 switch(iChoice) { case 1: openAccount(); //开户 break; case 2: cancelAccount(); //注销账户 break; case 3: save(); //存款 break; case 4: withdraw(); //取款 break; case 5: showAccount(); //查询余额 break; case 6: transferAccounts(); //转账 break; case 7: reportLoss(); //挂失 break; case 8: cancelLoss(); //解除挂失 break; case 9: updatePassword(); //更改密码 break; case 0: printf("欢迎您再来. \n"); } } while(iChoice); } /* 功能:从文件中读取储户数据 */ void readData() { FILE *fp; //用于文件操作 int i = 0; //从文件中取出余额 if ((fp=fopen("account.dat", "r"))==NULL) { printf("data file cannot open!"); exit(0); } while(fscanf(fp,"%d %s %d %lf %d",&user[i][0], name[i], &user[i][1], &balance[i],&user[i][2] ) != EOF) { i++; } N = i; //用全局变量存储用户的总人数 fclose(fp); return; } /* 功能:将用户数据保存到文件中 */ void writeData() { FILE *fp; //用于文件操作 int i=0; //保存余额 if ((fp=fopen("account.dat", "w"))==NULL) { printf("data file cannot open!"); exit(0); } for(i=0; i<N; i++) fprintf(fp,"%d %s %d %f %d\n",user[i][0], name[i], user[i][1], balance[i],user[i][2]); fclose(fp); } /* 功能:显示菜单并由业务员选择 返回值:用户选择的功能,范围0-9 */ int chooseInMenu() { int i; while(1) { printf("\n"); printf("+----------------------------+\n"); printf("+ 1 开户 2 销户 3 存款 +\n"); printf("+ 4 取款 5 查询 6 转账 +\n"); printf("+ 7 挂失 8 解挂 9 改密 +\n"); printf("+ 0 退出 +\n"); printf("+----------------------------+\n"); printf("请输入操作指令:"); scanf("%d", &i); if(i>=0 && i<=9) break; else printf("请重新选择功能\n\n"); } return i; } /* 功能:开户 说明:在进入系统时,在读入数据过程中,已经记录了用户数为N,在数组中对应下标为0~N-1 开户时要增加一个用户,只要为下标为N的数组元素置值,并在成功之后令N++即可。 这样做顺序增加简单,但遗留的后患是,在查询账户时,不得不用顺序查找,这在效率上是不划算的。 改进的手段(1):开户时,根据账号,将数据插入到数组中,使按账号有序,这样做插入时麻烦,但有利于以后要频繁的查询操作 改进的手段(2):账号由系统自动生成,保证其连续,这样在顺序增加的时候,就保证了其有序。 */ void openAccount() { if(N==upNum) { printf("银行用户数已经达到上限,不能再开户"); return; } //下面正常办理开户业务 printf("正在开户\n"); printf("账号:"); scanf("%d", &user[N][0]); printf("户主姓名:"); scanf("%s", name[N]); int iPass1, iPass2; printf("密码:"); iPass1=inputPassword(); //输入密码1 printf("确认密码:"); iPass2=inputPassword(); //输入密码2 if(iPass1==iPass2) { user[N][1]=iPass1; user[N][2]=0; //账户状态为“正常” printf("存入金额:"); scanf("%lf", &balance[N]); N++; //正式用户数增加1,确认了新用户已经加入 printf("成功开户!\n"); } else { printf("两次密码不一致,未成功开户!\n"); //没有N++,则读入的值无效 } } /* 功能:注销账户 说明:找到账户,并将其状态改为2-注销即可。 注销前应该检查余额,应该先取款再注销 */ void cancelAccount() { int id; //用于输入的账号 int who; //查找到该账号在数组中对应的下标 int iPass; printf("待销户账号:"); scanf("%d", &id); who = search(id); //根据账号查询用户,返回用户的下标 if(who<0) //说明id账户不存在 { printf("该用户不存在,销户失败!\n"); } else { printf("户主姓名:%s\n", name[who]); printf("密码:"); iPass=inputPassword(); if(iPass==user[who][1]) { printf("余额:%.2f 元\n", balance[who]); printf("确认销户(y/n)?"); if(tolower(getchar())=='y') { balance[who]=0; //取款后余额变0 user[who][2]=2; //状态变为注销 printf("取款 %.2f 元,销户成功!\n", balance[who]); } else { printf("你取消了操作,销户失败!\n"); } fflush(stdin); //清除了getchar()时在键盘缓存中的遗留,以免影响后续操作 } else { printf("输入的密码错误,销户失败!\n"); } } } /* 功能:存款 说明:需要保证账户存在,且处于正常状态 */ void save() { int id, who; double money; printf("账号:"); scanf("%d", &id); who = search(id); //根据账号查询用户,返回用户的下标 if(who<0) //说明id账户不存在 { printf("该用户不存在,存款失败!\n"); } else { if(user[who][2]==0) { printf("户主姓名:%s\n", name[who]); printf("输入存款额:"); scanf("%lf", &money); balance[who]+=money; printf("存款后,您有%.2f元. \n",balance[who]); } else if(user[who][2]==1) { printf("该用户处于挂失状态,存款失败!\n"); } else { printf("该用户已经销户,存款失败!\n"); } } return; } /* 功能:取款 说明:需要保证账户存在,且处于正常状态,另外,余额要足够取 */ void withdraw() { int id, who; int iPass; double money; printf("账号:"); scanf("%d", &id); who = search(id); //根据账号查询用户,返回用户的下标 if(who<0) //说明id账户不存在 { printf("该用户不存在,取款失败!\n"); } else { if(user[who][2]==0) { printf("户主姓名:%s\n", name[who]); printf("密码:"); iPass=inputPassword(); if(iPass!=user[who][1]) { printf("输入密码错误,取款失败!\n"); } else { printf("输入取款额:"); scanf("%lf", &money); if(money>balance[who]) //亲,不玩透支 { printf("余额不足,取款失败!\n"); } else { balance[who]-=money; printf("取款后,还有%.2f元. \n",balance[who]); } } } else if(user[who][2]==1) { printf("该用户处于挂失状态,取款失败!\n"); } else { printf("该用户已经销户,取款失败!\n"); } } return; } /* 功能:查询账户 说明:显示账户信息 */ void showAccount() { int id, who; int iPass; printf("账号:"); scanf("%d", &id); who = search(id); //根据账号查询用户,返回用户的下标 if(who<0) //说明id账户不存在 { printf("该用户不存在,查询完毕!\n"); } else { printf("户主姓名:%s\n", name[who]); printf("密码:"); iPass=inputPassword(); if(iPass!=user[who][1]) { printf("输入密码错误,不能继续查询其他信息!\n"); } else { printf("余额:%.2f元. \n",balance[who]); printf("状态:"); if(user[who][2]==0) { printf("正常\n"); } else if(user[who][2]==1) { printf("挂失\n"); } else { printf("已经销户\n"); } } } return; } /* 功能:转账 说明:需要保证两个账户都存在,且处于正常状态,另外,转出账户的余额要足够 */ void transferAccounts() { int id, whoout, whoin; int iPass; double money; printf("输入转出账号:"); scanf("%d", &id); whoout = search(id); //根据账号查询用户,返回用户的下标 if(whoout<0) //说明id账户不存在 { printf("该用户不存在,转账失败!\n"); } else { if(user[whoout][2]==0) { printf("户主姓名:%s\n", name[whoout]); printf("密码:"); iPass=inputPassword(); if(iPass!=user[whoout][1]) { printf("输入密码错误,转账失败!\n"); } else { printf("输入转账金额:"); scanf("%lf", &money); if(money>balance[whoout]) //亲,不玩透支 { printf("余额不足,转账失败!\n"); } else { printf("输入转入账号:"); scanf("%d", &id); whoin = search(id); //根据账号查询用户,返回用户的下标 if(whoin<0) { printf("转入账户不存在,转账失败!\n"); } else { if(user[whoin][2]>0) { printf("转入账户异常,转账失败!\n"); } else { balance[whoout]-=money; balance[whoin]+=money; printf("取款后,您还有%.2f元. \n",balance[whoout]); } } } } } else { printf("该账户异常,转账失败!\n"); } } return; } /* 功能:挂失账户 */ void reportLoss() { int id, who; int iPass; printf("账号:"); scanf("%d", &id); who = search(id); //根据账号查询用户,返回用户的下标 if(who<0) //说明id账户不存在 { printf("该用户不存在,不能挂失!\n"); } else { printf("户主姓名:%s\n", name[who]); printf("密码:"); iPass=inputPassword(); if(iPass!=user[who][1]) { printf("输入密码错误,不能继续操作!\n"); } else { if(user[who][2]==0) { user[who][2]=1; printf("挂失成功\n"); } else if(user[who][2]==1) { printf("该账户已经处于挂失状态\n"); } else { printf("该账户已销户,不能挂失\n"); } } } return; } /* 功能:解除挂失 */ void cancelLoss() { int id, who; int iPass; printf("账号:"); scanf("%d", &id); who = search(id); //根据账号查询用户,返回用户的下标 if(who<0) //说明id账户不存在 { printf("该用户不存在,解除挂失失败!\n"); } else { printf("户主姓名:%s\n", name[who]); printf("密码:"); iPass=inputPassword(); if(iPass!=user[who][1]) { printf("输入密码错误,不能继续操作!\n"); } else { if(user[who][2]==0) { printf("该账户处于正常状态,不需要解除挂失\n"); } else if(user[who][2]==1) { user[who][2]=0; printf("解除挂失成功\n"); } else { printf("该账户已销户,操作无效\n"); } } } return; } /* 功能:改密码 */ void updatePassword() { int id, who; int iPass, iPass1, iPass2; printf("账号:"); scanf("%d", &id); who = search(id); //根据账号查询用户,返回用户的下标 if(who<0) //说明id账户不存在 { printf("该用户不存在,修改密码失败!\n"); } else { printf("户主姓名:%s\n", name[who]); printf("密码:"); iPass=inputPassword(); if(iPass!=user[who][1]) { printf("输入密码错误,不能继续操作!\n"); } else { printf("新密码:"); iPass1=inputPassword(); //输入密码1 printf("确认密码:"); iPass2=inputPassword(); //输入密码2 if(iPass1==iPass2) { user[who][1]=iPass1; printf("修改成功!\n"); } else { printf("两次输入不同,修改失败!\n"); } } } return; } /* 功能:输入密码 返回值:整型的密码值 技术说明: (1)此功能在多个模块中都要用到且功能单一,故分离出来,单独作业一个函数 (2)为了便于在输入中只显示*,接受输入时以字符形式输入,而后转为对应的整型数 (3)规定密码由不全为0的6位数字构成(当开头是'0'时,实际不足6位),一旦输入错误将重新输入 附:在实际的系统中,密码通常用字符串描述,即使只允许出现数字字符 */ int inputPassword() { char ch; //接收字符形式密码 int iPass=0; //要转换为数字 int i; while(1) { for(i=0; i<6; i++) { ch=getch(); //输入但不显示 putchar('*'); //输出* if(isdigit(ch)) iPass=iPass*10+(ch-'0'); else { iPass=0; break; //退出for循环后,再次接受 } } fflush(stdin); //清除键盘缓存区中已经有的输入 printf("\n"); if(iPass==0) //此条件成立可能由两种情况引起:输入了非数字字符被直接重置为0,或6位全0后正常退出for循环 { printf("密码要求全为数字,且不能全0!\n"); printf("请重新输入密码: "); } else break; } return iPass; } /* 功能:根据账号查询用户,返回用户的下标 入口参数:要查询用户的账号 返回值:如果该用户存在,返回该用户在数组中的下标,否则,返回一个负数(-1) 说明: 由于不能保证在user数组中按账号有序,本模块不得不采用顺序查找,这是一个效率很低的算法 如果在开户模块中保证了按账号有序,本函数可以选择更快的算法(见开户模块的说明) */ int search(int id) { int index=-1; int i; for(i=0; i<N; i++) { if(user[i][0]==id) { index=i; break; //找到了,立即退出循环 } } return index; //若找到,其值在0~N-1间,否则,保持-1 }