查找文本文件中的关键字

查找文本文件中的关键字,说白了就是以文本文件作为输入,进行字符串匹配,找返回其第一次出现的下标位置。但是由于数据是以文本文件的形式作为输入的,如何存储和进行匹配就成为了一个问题。下面以两种方法来介绍如何操作。注:本文中采用的字符串匹配算法只是普通的字符串匹配算法,重点在对文件处理和分块查找。

一、蛮力法
这种方法非常简单,把文件中的所有数据输入到一个字符数组中,然后以数组作为主串,关键字为模式串,进行字符串匹配即可。

但是这里有一个问题,就是字符数组要多大才合适?由于不同的文件的数据量可能差别非常大,所以我们应该根据文件的大小来动态分配字符数组来存储主串。即我们现在的问题变为如何获得文件的大小。文件的大小可以用如下的方法来获得,首先打开文本文件,保存其文件位置,然后把文件指针定位到文件的末尾,获得其偏移量,然后再把文件指针恢复到原先即可。恢复文件指针是为了不让该调用对文件的其他操作产生影响,从外部看来这个操作调用前与调用后文件的状态并没有变化过。其现实代码如下,返回文件所占的字符总数:
[cpp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. int GetFileLength(ifstream &inputFile)  
  2. {  
  3.     //保存文件当前位置  
  4.     streampos pos = inputFile.tellg();  
  5.     //定位到文件尾  
  6.     inputFile.seekg(0, ios::end);  
  7.     //返回文件尾的偏移量,即文件的大小  
  8.     int length = inputFile.tellg();  
  9.     //返回到文件先前的位置  
  10.     inputFile.seekg(pos);  
  11.     return length;  
  12. }  

则实现字符串匹配的函数如下:
[cpp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. int IndexInFile(const char *fileName, const char *keyWord)  
  2. {  
  3.     //以只读方式,打开文件fileName  
  4.     ifstream inputFile(fileName);  
  5.     if(!inputFile)  
  6.     {  
  7.         //打开文件失败  
  8.         cerr<<"error: unable to open input file: "  
  9.             <<fileName<<endl;  
  10.         return -1;  
  11.     }  
  12.     //获得文件的长度,即字节数,并开劈一个同样大小的数组保存文件数据  
  13.     int length = GetFileLength(inputFile);  
  14.     char *text = new char[length+1];  
  15.     //把文件的内容讲到数组中  
  16.     inputFile.read(text, length);  
  17.     inputFile.close();  
  18.     text[length] = '\0';  
  19.     //进行模式串匹配,并返回结果  
  20.     int index = IndexOf(text, length, keyWord, strlen(keyWord));  
  21.     delete []text;  
  22.     return index;  
  23. }  
  24. int IndexOf(const char *text, int textSize,  
  25.             const char *match, int matchSize)  
  26. {  
  27.     for(int i = 0; i <= textSize - matchSize; ++i)  
  28.     {  
  29.         int j = 0;  
  30.         while(j < matchSize && match[j] == text[i+j])  
  31.             ++j;  
  32.         //所有的字符都与文本中的一致,则匹配成功  
  33.         if(j == matchSize)  
  34.             return i;  
  35.     }  
  36.     //匹配失败  
  37.     return -1;  
  38. }  
其代码非常简单,不再多说了。

二、分治法
注意,这里主要是用到了把文件分成若干大小相同的块,并对各个块进行字符串匹配的方法来处理,并不是指字符串匹配算法使用了分治的思想。

由于文件的数据可能非常巨大,一次性地把文件的所有内容读入到内存时,有时是不可能的,而且这样做也没有什么必要,因为我们要查找的串很可能就在文件的前面部分,而我们却把一个文件的所有内容调入到内存中,浪费了大量的内存空间,而且效率不高。所以我们应该把文件进行分块处理,每次从文件中读取一定的字符到缓冲区中,进行处理。其实现方法如下:

首先把开文件,每次从文件读取指定个数的字符到buffer中,然后以buffer中的字符作为主串,关键字作为模式串进行字符串匹配,若匹配成功把返回其下标,若匹配不成功,则把下标值累加上读到缓冲区中字符的个数,并继续从文件中读取字符到buffer中,继续对buffer进行字符串匹配,直到找到关键字,或文件结束。

这个方法不用把文件的所有内容调入内存中,而是每次都从内存读入一个buffer的内容,可以减少内存的开销,不论文件有多大都可行。然而这个方法的难点在哪里呢,难点就要当关键字在两个缓冲区之间时该如何识别和处理。

下面说说我的想法,为了方便解说,我们假设buffer的大小为6,要查找的关键字为defg,文件的内容为abcdefghijk,我们每次从文件中读取5个字符到buffer中(第6个字符为'\0'),为abcde,可以看到我们要查找的字符串只有一部分在buffer中,它们被分为了两个部分。首先我们判断当在buffer上查找不成功,是否是由于所有的字符都匹配,但是模式串还没匹配完,主串却已经到了尽头,若是,把把之前与模式串匹配的部分字符复杂到buffer的前面,然后再从文件中输入数据,并放到其后面。在这个例子中,就是把de复制到buffer的前面,再从文件中读取数据到de后面的buffer中,读入完毕后,buffer的数据变成defgh,然后再对其进行匹配,即可匹配成功。

其实现代码如下:
[cpp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. int IndexInFile(const char *fileName, const char *keyWord)  
  2. {  
  3.     //以只读方式,打开文件fileName  
  4.     ifstream inputFile(fileName);  
  5.     if(!inputFile)  
  6.     {  
  7.         //打开文件失败  
  8.         cerr<<"error: unable to open input file: "  
  9.             <<fileName<<endl;  
  10.         return -1;  
  11.     }  
  12.     int keySize = strlen(keyWord);//关键字的长度  
  13.     int index = 0; //记录关键字首次出现的位置  
  14.     int lastMatch = 0;//记录最后一次匹配的位置  
  15.     const int bufferSize = 5;  
  16.     char buffer[bufferSize + 1];  
  17.     while(!inputFile.eof())  
  18.     {  
  19.         //把最后一次比较配置的字符复制到最前面  
  20.         WriteEndToBegin(buffer, bufferSize, lastMatch);  
  21.         //读入数据到缓冲区中lastMatch后的位置中  
  22.         inputFile.read(buffer + lastMatch,  
  23.                        bufferSize - lastMatch);  
  24.         buffer[bufferSize] = '\0';  
  25.         cout<<buffer<<endl;  
  26.         int thisIndex = IndexOf(buffer, bufferSize,  
  27.                                 keyWord, keySize, lastMatch);  
  28.         if(thisIndex != -1)  
  29.         {  
  30.             //若查找成功,则下标值为之前查找过的字符数加上此次查找的字符数  
  31.             index += thisIndex;  
  32.             return index;  
  33.         }  
  34.         else  
  35.         {  
  36.             //若查找不成功,则加上新放入到缓冲区的字符数  
  37.             index += (bufferSize - lastMatch);  
  38.         }  
  39.     }  
  40.     return -1;  
  41. }  
  42. int IndexOf(const char *text, int textSize,  
  43.             const char *match, int matchSize,  
  44.             int &lastMatch)  
  45. {  
  46.     for(int i = 0; i < textSize; ++i)  
  47.     {  
  48.         lastMatch = 0;  
  49.         while(lastMatch < matchSize && match[lastMatch] == text[i+lastMatch])  
  50.             ++lastMatch;  
  51.         //所有的字符都与文本中的一致,则匹配成功  
  52.         if(lastMatch == matchSize)  
  53.             return i;  
  54.         //所有的字符都匹配,但是模式串还没匹配完,主串已经到了尽头  
  55.         if(i + lastMatch == textSize)  
  56.             break;  
  57.     }  
  58.     //匹配失败  
  59.     return -1;  
  60. }  
  61. void WriteEndToBegin(char *text, int textSize, int writeCount)  
  62. {  
  63.     //把字符数组中最后writeCount个字符写到最前面  
  64.     for(int i = 0; i < writeCount; ++i)  
  65.     {  
  66.         text[i] = text[textSize - writeCount + i];  
  67.     }  
  68. }  

代码分析:
这里主要解析一下IndexOf函数中的lastMatch参数,该参数记录了主串中与模式串匹配的字符的个数。当IndexOf函数返回-1时,它尤其有用,因为它让我们知道,在buffer中,有多少个字符已经与模式串(关键字)匹配了,在上面的例子中,就是de两个字符,则其值为2,所以把buffer中最后的两个字符复制到了其前面。

同时还要 注意,buffer的大小一定要大小模式串的长度,不然的话,buffer会因为装不下一个模式串而出错。在代码中为了演示而把buffer的大小定为6(5+1),但是在使用时应该把它改变成你想要的大小,通常256或512是一个合适的值。

三、复杂度分析
两个算法的时间复杂度无为O(n*m),n为文件的字符数,m为关键字的字符数(即模式串的字符数)。空间复杂度第一个算法为O(n),第二个算法为O(1),因为不论文件多大,缓冲区的大小都是确定的。

源代码可以点击这里下载:


PS:本人成为了CSDN2013博客之星候选人之一,如果你觉得本人写的博客还可以,请投我一票,支持一下我,我的投票地址是:
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值