首页 > 代码库 > 用数组实现从文件搜索帐户和验证密码
用数组实现从文件搜索帐户和验证密码
最近一个同学在写个银行管理系统,然后问我怎么从文件搜索帐户,给了方法后又不懂文件里面的密码怎么与输入的匹配。一般来说,高效的做法是用链表实现。用数组实现不是高效的方法,而且浪费空间。再者,对于账户类有个人信息集合的,一般用结构体,代码写起来也方便简单。
但是他却用数组来做,而且没有用结构体。对于这种情况下如何搜索帐户,如何验证密码呢? 我尝试了一下,发现不难解决。
解决这个问题的方法:用文件数据的排序定位来做。
第一步:文件信息读入
用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] = {‘\0‘}; //输入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运行
初步完成了这个功能,现在把完整代码贴出来。
#include <iostream>#include <cstdlib>#include <cstring>#include <windows.h>using namespace std;//检测密码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;}int main(){ char a[20]; char shuRu[20] = {‘\0‘}; int flag = 0; cout << "请输入账号名: "; cin >> shuRu; FILE* fp; fp = fopen("test.txt","rb"); while(fscanf(fp,"%s",a) > 0) { if (strcmp(shuRu,a) == 0) { flag = 1; fscanf(fp,"%s",a); fscanf(fp,"%s",a); break; } } if (flag == 0) cout << "用户不存在,请注册!" << endl; else checkKey(a); fclose(fp); return 0;}
我们来运行一下。
首先文件数据如下:(每个帐户的第三行为帐号,第五行为密码)
运行结果:
第六步:bug修复
看上去好像完成了相应预期功能,但是细心观察,不难发现一个bug。举例说明一下:
当文件内容为:
可以看出,第一个帐户的密码和第二个帐户的账号相同,都是aabbcc,此时程序运行就有错误,当搜索到了aabbcc,程序就把他当成了账号,于是出错。
修复bug也很简单,用一种特殊字符对账号进行处理。比如在账号后面追加一个 “@”。
因此,程序要修改两个地方
1. 帐户注册后把信息写入文件时,在账号后面追加一个 “@”
2. 登录时,当输入账号完毕后,也给输入的账号后面追加一个“@”
第一个地方不是我们讨论的范围,我们来看看与本篇随笔有关的要修改的第二个地方:
相关函数: strcat(str1,str2);实现把字符数组 str2 追加到 str1 后面。
代码如下:
cout << "请输入账号名: ";cin >> shuRu;strcat(shuRu,"@");
所以,当你登录时,输入账号完成按回车键时,程序会自动给你输入的账号后面追加一个字符”@“,然后再与文件数据进行比较。
修改后的完整代码如下:
#include <iostream>#include <cstdlib>#include <cstring>#include <windows.h>using namespace std;//检测密码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;}int main(){ char a[20]; char shuRu[20] = {‘\0‘}; int flag = 0; cout << "请输入账号名: "; cin >> shuRu; strcat(shuRu,"@"); FILE* fp; fp = fopen("test.txt","rb"); while(fscanf(fp,"%s",a) > 0) { if (strcmp(shuRu,a) == 0) { flag = 1; fscanf(fp,"%s",a); fscanf(fp,"%s",a); break; } } if (flag == 0) cout << "用户不存在,请注册!" << endl; else checkKey(a); fclose(fp); return 0;}
文件数据:
运行结果:
密码匹配成功。
总结:对于没有用链表 + 结构体的来写帐户登记的程序,属于纯文件信息处理。那么就只能差强人意的用一些方法来解决。用的是”排序定位“的方法。
代码是追求高效,简洁的。一开始没有用好的方法去解决,虽然也能通向罗马,从程序的维护和更新的角度来看,是不推荐的。