zoukankan      html  css  js  c++  java
  • 用数组实现从文件搜索帐户和验证密码

    最近一个同学在写个银行管理程序(C语言小项目),然后问我怎么从文件搜索帐户,给了方法后又不懂文件里面的密码怎么与输入的匹配。一般来说,高效的做法是用链表实现。用数组实现不是高效的方法,而且浪费空间。再者,对于账户类有个人信息集合的,一般用结构体,代码写起来也方便简单。

    但是他却用数组来做,而且没有用结构体。对于这种情况下如何搜索帐户,如何验证密码呢? 我尝试了一下,发现不难解决。

    解决这个问题的方法:用文件数据的排序定位来做。

    第一步:文件信息读入

     用fscanf 实现文件读出文件信息。

    1 char a[20];
    2 
    3 FILE* fp;
    4 fp = fopen("test.txt","rb");
    5 fscanf (fp, "%s", a);
    6 fclose(fp);

    fcanf读取数据,以空格为分割点。比如对于文件内容为“abc ABC”(双引号之内)的情况,用如下代码:

    fscanf (fp, "%s", a);
    fscanf (fp, "%s", b);

     得到字符数组a 为abc;字符数组b为 ABC;

    若文件内容为“abcABC”(双引号之内)的情况,即abc和ABC之间没有空格隔开,用以上代码,得到的结果将是:

    字符数组a 为abcABC;字符数组b 为乱码。

    fscanf();还有一个特点,就是在一个程序里面是顺序读入的。

    在此举个例子:

    txt文件(每个数据一行)内容:

    abc

    ABC

    ruby

    然后执行下面的代码:

    int main()
    {
      char a[20],b[20],c[20];
      FILE* fp;
      fp = fopen("test.txt","rb");
      fscanf(fp, "%s", a);
      //……此处省略n行代码
      fscanf(fp, "%s",b);
      //……此处省略n行代码
     fscanf(fp, "%s",c);
     fclose(fp);
    return 0;
    }

     得到的结果是:

    字符数组a为 abc,b为ABC,c为ruby;

    第二步:搜索账号

    账号搜索的方法是遍历文件数据,找到与输入匹配的账号就停止搜索。

    用 while(fscanf(fp,"%s",a) > 0) 实现遍历文件数据。

    作用是把文件内容一行一行读入赋值给字符数组a,然后再与输入的账号进行比较。

    同时使用标记 flag 判断是否找到匹配的账号,以便后续处理各种不同情况。

    代码如下:

    char a[20];
    char shuRu[20] = {''}; //输入
    
    int flag = 0;
    
    cout << "请输入账号名: ";
    cin >> shuRu;
    
    FILE* fp;
    fp = fopen("test.txt","rb");
    
    while(fscanf(fp,"%s",a) > 0) //遍历文件数据
    {
        if (strcmp(shuRu,a) == 0)
        {
                    cout << "找到匹配账号" << endl;
            flag = 1;
                    break;
            }
    }
    
    if (flag == 0){
      cout << "用户不存在,请注册!" << endl;
      //下一步
    }
    else
      //下一步;
    fclose(fp);

     代码中判断数据相同用strcmp(str1,str2); 如果两个字符数组存储的内容相同,则 strcmp(str1,str2)== 0

    此时停止搜索,用break;跳出循环。

    现在举例一下:

    其中第三行为账号,第五行为密码。其他的为姓名,地址,年龄等其他信息。

    运行如下:

     第三步:定位文件密码数据

    如果是用结构体,当检测到帐号的时候,再用结构体的 “.” 也就是 “点”密码 来解决。简单方便。但是用的不是结构体,所以只能用其他方法。随笔开头写了用“排序定位”来做,如何实现?

    从这个文件数据来看,账号与密码分别是第三行, 第五行,中间隔了个第四行。下面另一个帐户也是同样的排序。

    那么就定位到第五行,然后再进行 输入密码 与 文件数据的比较。

    代码如下:

    if (strcmp(shuRu,a) == 0)
    {
        flag = 1;
        fscanf(fp,"%s",a);
        fscanf(fp,"%s",a);
        break;
    }

     这里面当检测到帐号的时 flag = 1;表示找到匹配账号。

    然后用了 两个fscanf(fp,"%s",a); 这不是代码错误,前面提到了fscanf(); 是顺序读入,并且举了 a,b,c三个字符数组的例子。

    这里再说明一下为何用两个fscanf();

    第一个fscanf();是把账号下面的第一个数据赋值给了 字符数组a;

    第二个fscanf();是把账号下面的第二个数据赋值给了 字符数组a;

    由于密码数据是账号数据下面的第二个数据,所以必须用两个fscanf(); 因为fscanf();为顺序读入,无法进行跳跃。

    第四步:验证密码  

    定位了密码数据,那么就可以进行密码验证了。为了实现当密码输入错误时,能重新输入,我们必须把验证密码这个环节写成一个函数,然后自身循环调用,类似递归的用法。

    代码如下:

    void checkKey(char a[20])
    {
        char mima[20];
        cout << "输入密码:";
        cin >> mima;
        if (strcmp(a, mima) == 0)
            cout << "登录成功" << endl;
        else
        {
            cout << "密码输入错误!请重新输入。" << endl;
            Sleep(2000);
            checkKey(a);
        }
    
        return;
    }

     第五步:demo运行

    初步完成了这个功能,现在把完整代码贴出来。

     1 #include <iostream>
     2 #include <cstdlib>
     3 #include <cstring>
     4 #include <windows.h>
     5 using namespace std;
     6 
     7 //检测密码
     8 void checkKey(char a[20])
     9 {
    10     char mima[20];
    11     cout << "输入密码:";
    12     cin >> mima;
    13     if (strcmp(a, mima) == 0)
    14         cout << "登录成功" << endl;
    15     else
    16     {
    17         cout << "密码输入错误!请重新输入。" << endl;
    18         Sleep(2000);
    19         checkKey(a);
    20     }
    21 
    22     return;
    23 }
    24 
    25 int main()
    26 {
    27     char a[20];
    28     char shuRu[20] = {''};
    29 
    30     int flag = 0;
    31 
    32     cout << "请输入账号名: ";
    33     cin >> shuRu;
    34 
    35     FILE* fp;
    36     fp = fopen("test.txt","rb");
    37 
    38     while(fscanf(fp,"%s",a) > 0)
    39     {
    40 
    41         if (strcmp(shuRu,a) == 0)
    42         {
    43             flag = 1;
    44             fscanf(fp,"%s",a);
    45             fscanf(fp,"%s",a);
    46             break;
    47 
    48         }
    49         
    50     }
    51     if (flag == 0)
    52         cout << "用户不存在,请注册!" << endl;
    53     else
    54         checkKey(a);
    55 
    56     fclose(fp);
    57     return 0;
    58 }

    我们来运行一下。

    首先文件数据如下:(每个帐户的第三行为帐号,第五行为密码)

    运行结果:

    第六步:bug修复 

    看上去好像完成了相应预期功能,但是细心观察,不难发现一个bug。举例说明一下:

    当文件内容为:

    可以看出,第一个帐户的密码和第二个帐户的账号相同,都是aabbcc,此时程序运行就有错误,当搜索到了aabbcc,程序就把他当成了账号,于是出错。

    修复bug也很简单,用一种特殊字符对账号进行处理。比如在账号后面追加一个 “@”。

    因此,程序要修改两个地方

    1. 帐户注册后把信息写入文件时,在账号后面追加一个 “@”

    2. 登录时,当输入账号完毕后,也给输入的账号后面追加一个“@”

    第一个地方不是我们讨论的范围,我们来看看与本篇随笔有关的要修改的第二个地方:

    相关函数: strcat(str1,str2);实现把字符数组 str2 追加到 str1 后面。

    代码如下:

    cout << "请输入账号名: ";
    cin >> shuRu;
    strcat(shuRu,"@");

     所以,当你登录时,输入账号完成按回车键时,程序会自动给你输入的账号后面追加一个字符”@“,然后再与文件数据进行比较。

    修改后的完整代码如下:

     1 #include <iostream>
     2 #include <cstdlib>
     3 #include <cstring>
     4 #include <windows.h>
     5 using namespace std;
     6 
     7 //检测密码
     8 void checkKey(char a[20])
     9 {
    10     char mima[20];
    11     cout << "输入密码:";
    12     cin >> mima;
    13     if (strcmp(a, mima) == 0)
    14         cout << "登录成功" << endl;
    15     else
    16     {
    17         cout << "密码输入错误!请重新输入。" << endl;
    18         Sleep(2000);
    19         checkKey(a);
    20     }
    21 
    22     return;
    23 }
    24 
    25 int main()
    26 {
    27     char a[20];
    28     char shuRu[20] = {''};
    29 
    30     int flag = 0;
    31 
    32     cout << "请输入账号名: ";
    33     cin >> shuRu;
    34     strcat(shuRu,"@");
    35 
    36     FILE* fp;
    37     fp = fopen("test.txt","rb");
    38 
    39     while(fscanf(fp,"%s",a) > 0)
    40     {
    41 
    42         if (strcmp(shuRu,a) == 0)
    43         {
    44             flag = 1;
    45             fscanf(fp,"%s",a);
    46             fscanf(fp,"%s",a);
    47             break;
    48 
    49         }
    50         
    51     }
    52     if (flag == 0)
    53         cout << "用户不存在,请注册!" << endl;
    54         
    55     else
    56         checkKey(a);
    57 
    58     fclose(fp);
    59     return 0;
    60 }

     文件数据:

    运行结果:

    密码匹配成功。

    总结:对于没有用链表 + 结构体的来写帐户登记的程序,属于纯文件信息处理。那么就只能差强人意的用一些方法来解决。用的是”排序定位“的方法。

    代码是追求高效,简洁的。一开始没有用好的方法去解决,虽然也能通向罗马,从程序的维护和更新的角度来看,是不推荐的。

  • 相关阅读:
    数组字典
    字符串创建
    不可变数组
    数组
    纯净控制台
    可变字典
    数组Array
    id NSObject instanceType 区别
    可变字符串 插入,删除,替换,赋值
    查询字符串中某部分的字符串
  • 原文地址:https://www.cnblogs.com/hlwyfeng/p/3900323.html
Copyright © 2011-2022 走看看