C语言提高-40讲: 小小型应用系统开发指导(四)(银行储蓄系统数组版)

戳原创

把用户信息及实际用户数目作为全局变量
用户信息包括:账号、用户名、密码、余额、用户状态(“正常”“挂失”“已销户”)
主函数包含:业务登录、读取储户数据、业务驱动、保存数据
业务驱动模块包含:开户、注销、存款、取款、查询、转账、挂失、解除挂失、更改密码
所有子模块,均需要验证输入密码;除开户以外,都需要进行查找用户
每一次在选择业务前都需要显示菜单并由业务员选择业务

储户信息的存储:
账号、密码、用户状态均可以用整型值表示
定义二维数组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();
代码描述:调用查找模块,找到后显示户主姓名,验证密码通过后,输入并确认新密码,
通过后,将新密码覆盖到用户信息对应密码数组的该用户上


worker.txt   account.dat

#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
来源:知乎
著作权归作者所有。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值