戳原创:
把用户信息及实际用户数目作为全局变量
用户信息包括:账号、用户名、密码、余额、用户状态(“正常”“挂失”“已销户”)
主函数包含:业务登录、读取储户数据、业务驱动、保存数据
业务驱动模块包含:开户、注销、存款、取款、查询、转账、挂失、解除挂失、更改密码
所有子模块,均需要验证输入密码;除开户以外,都需要进行查找用户
每一次在选择业务前都需要显示菜单并由业务员选择业务
储户信息的存储:
账号、密码、用户状态均可以用整型值表示
定义二维数组int user[upNum][3];表示每一位储户的其中三个整型信息
且user[upNum][2]=0、1、2 分别表示“正常”“挂失”“已销户”三种状态
用户名用二维字符串数组表示
账户余额用double类型的一维数组表示
二维数组的行下标和一维数组的下标均相同,表示同一个人的信息
业务登录模块 int pass();
新建文件worker.txt,输入业务员名,登录密码,用空格或换行隔开
从文件中读入数据并分别保存到数组sNameInFile, sPassInFile中
输入密码:将读入的字符放到数组sPass中,每次接受到的字符在屏幕上只显示*
使用getch()后,sPass已存入密码,清除缓存
如果登录名和密码均正确,返回值为1,回到主程序中继续下一步运行
总共可以进行3次尝试,超过3次,不能进入系统,并打印出来
读取储户数据模块 void readData();
扫描文件里的储户信息到对应数组中,并记录当前总人数
保存储户数据模块 void writeData();
将更新后所有储户的信息保存到文件中
业务驱动模块 void work();
从选择菜单模块中获得操作指令,并将所有业务办理的接口写入多分支语句中
菜单上的操作指令与业务办理的选择条件一一对应,根据不同的操作指令实现不同的业务办理
选择菜单模块 int chooseInMenu();
打印菜单界面,并将操作指令作为返回值返回到业务驱动模块中
查找模块 int search(int id);
功能:根据账号查询用户,返回用户的下标
入口参数:要查询用户的账号
返回值:如果该用户存在,返回该用户在数组中的下标,否则,返回一个负数(-1)
代码描述:即比对输入的密码是否能在存入的账号数组里找到
由于不能保证在user数组中按账号有序,本模块不得不采用顺序查找
输入密码模块 int inputPassword();
功能:输入密码
返回值:整型的密码值
技术说明:
(1)为了便于在输入中只显示*,接受输入时以字符形式输入,而后转为对应的整型数
(2)规定密码由不全为0的6位数字构成,一旦输入错误将重新输入
代码描述:需要增加字符函数isdigit()用来判断输入的字符是否为数字,
输入非数字字符或全为0的字符,要求重新输入,将正确格式的密码作为返回值
开户模块 void work();
说明:在进入系统时,在读入数据过程中,已经记录了用户数为N,在数组中对应下标为0~N-1
开户时要增加一个用户,只要为下标为N的数组元素置值,并在成功之后令N++即可。
此处也说明了将实际的用户数目N作为全局变量的好处(N是变量,在某些模块中需要传递)
代码描述:N应小于人数上限,否则提示不能开户。将储户信息记入对应数组,账户状态为“正常”
注销模块 void cancelAccount();
说明:找到账户,并将其状态改为2-注销即可。注销前应该检查余额,应该先取款再注销
代码描述:调用查找模块,找到后显示户主姓名,验证密码,取走余额,更换状态
存款模块 void save();
说明:需要保证账户存在,且处于正常状态
代码描述:调用查找模块,找到后只有在正常状态下才能存款,否则打印相应状态下的内容
取款模块 void withdraw();
说明:需要保证账户存在,且处于正常状态,另外,余额要足够取
代码描述:调用查找模块,找到后验证是否在正常状态,是则打印户主姓名,验证密码,取钱
查询模块 void showAccount();
说明:显示账户信息
代码描述:调用查找模块,显示用户姓名,验证密码通过后,显示余额,打印状态
转账模块 void transferAccounts();
说明:需要保证两个账户都存在,且处于正常状态,另外,转出账户的余额要足够
代码描述:调用查找模块,查找转出用户,在状态正常下,验证密码,转帐
在余额大于转账额的情况下,查找并验证转入用户的状态是否正常,变动两个用户的余额
挂失模块 void reportLoss();
代码描述:调用查找模块,找到后显示户主姓名,验证密码通过后,检查用户状态,
在正常状态下更换为挂失,其余打印对应状态信息
解除挂失模块 void cancelLoss();
代码描述:调用查找模块,找到后显示户主姓名,验证密码通过后,检查用户状态,
在挂失状态下更换为正常,其余打印对应状态信息
更换密码模块 void updatePassword();
代码描述:调用查找模块,找到后显示户主姓名,验证密码通过后,输入并确认新密码,
通过后,将新密码覆盖到用户信息对应密码数组的该用户上
#include <stdio.h>
#include <conio.h> //包含getch()的头文件
#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')
{
printf("取款 %.2f 元,销户成功!\n", balance[who]);
balance[who]=0; //取款后余额变0
user[who][2]=2; //状态变为注销
}
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
}
PS:
A:为什么C语言中敲击回车键时getch返回的是'\r'而不是'\n'?
相对的反倒是Ctrl+J返回的是'\n'这是为什么呢?
Q:'\r' 回车 ASCII码13;'\n' 换行 ASCII码10
getch()函数是返回你按下的键的键值,它会设置控制台(黑色的cmd窗口)为无回显、无缓冲模式,
当按了一个键时,getch()函数会立刻返回键值(无缓冲),不会显示你输入的字符(无回显)。
scanf()、getchar()之类的函数,使用的是标准输入模式,按什么键,屏幕上就会显示相应字符,
当按回车键时,控制台会将输入的回车换成换行符,存入输入缓冲区,让这些函数取出来。
按了什么键, 输入了什么字符,输出了什么内容,主要是由控制台的工作模式决定。
至于Ctrl+J返回的是'\n',这是控制台提供的一种直接输入换行的方法。
作者:知乎用户
链接:https://www.zhihu.com/question/21526785/answer/18766531
来源:知乎
著作权归作者所有。