首页 > 代码库 > 用数组实现从文件搜索帐户和验证密码

用数组实现从文件搜索帐户和验证密码

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

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

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

第一步:文件信息读入                                            

 用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;}

 文件数据:

运行结果:

密码匹配成功。

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

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