zoukankan      html  css  js  c++  java
  • KMP 与 BM 算法实现

    #include <iostream>
    #include <vector>
    #include <string>
    #include <algorithm>
    using namespace std;
    //KMP
    vector<int> Process(string a)
    {
            int n = a.size();
            vector<int> f(n);
            f[0] = 0;
            int t = -1;
            //fuck!!!!!!!!!!!!!!!!!!!!!!!
            // I will remember this shit for ever!!!
            // t is the position minus 1
            for(int s = 0; s<n-1;++s)
            {
                    while(t > -1 && a[s+1]!= a[t+1])
                            t = f[t]-1;
                    if(a[s+1]==a[t+1]){
                            t = t+1;
                            f[s+1] = t+1;
                    }
                    else{
                            f[s+1] = 0;
                    }
                    //cout <<s+1 <<"   "<<f[s+1]<<endl;
            }
            return f;
    }
    
    // b is the compare string
    void KMP(string a,string b)
    {
            int m = a.size();
            int n = b.size();
            vector<int> P = Process(b);
            int j = 0;
            for(int i =0;i<m;++i)
            {
                    while( j>0 && a[i]!=b[j]) j = P[j-1]+1;
                    if(a[i]==b[j]) j=j+1;
                    if(j==n){
                            cout << "find at a"<<i<<endl;
                            j = P[j-1]+1;
                           //return;
                    }
            }
    
    }
    //////////////////////////////////////////////////////
    /*
        函数:int* MakeSkip(char *, int)
    	目的:根据坏字符规则做预处理,建立一张坏字符表
    	参数:
    		ptrn => 模式串P
    		PLen => 模式串P长度
        返回:
    	    int* - 坏字符表
    */
    int* MakeSkip(char *ptrn, int pLen)
    {	
    	int i;
    	//为建立坏字符表,申请256个int的空间
    	/*PS:之所以要申请256个,是因为一个字符是8位,
    	  所以字符可能有2的8次方即256种不同情况*/
    	int *skip = (int*)malloc(256*sizeof(int));
    
    	if(skip == NULL)
    	{
    		fprintf(stderr, "malloc failed!");
    		return 0;
    	}	
    
    	//初始化坏字符表,256个单元全部初始化为pLen
    	for(i = 0; i < 256; i++)
    	{
    		*(skip+i) = pLen;
    	}
    
    	//给表中需要赋值的单元赋值,不在模式串中出现的字符就不用再赋值了
    	while(pLen != 0)
    	{
    		*(skip+(unsigned char)*ptrn++) = pLen--;
    	}
    
    	return skip;
    }
    
    
    /*
    	函数:int* MakeShift(char *, int)
    	目的:根据好后缀规则做预处理,建立一张好后缀表
    	参数:
    		ptrn => 模式串P
    		PLen => 模式串P长度
        返回:
    	    int* - 好后缀表
    */
    int* MakeShift(char* ptrn,int pLen)
    {
    	//为好后缀表申请pLen个int的空间
    	int *shift = (int*)malloc(pLen*sizeof(int));
    	int *sptr = shift + pLen - 1;//方便给好后缀表进行赋值的指标
    	char *pptr = ptrn + pLen - 1;//记录好后缀表边界位置的指标
    	char c;
    
    	if(shift == NULL)
    	{
    		fprintf(stderr,"malloc failed!");
    		return 0;
    	}
    
    	c = *(ptrn + pLen - 1);//保存模式串中最后一个字符,因为要反复用到它
    
    	*sptr = 1;//以最后一个字符为边界时,确定移动1的距离
    
    	pptr--;//边界移动到倒数第二个字符(这句是我自己加上去的,因为我总觉得不加上去会有BUG,大家试试“abcdd”的情况,即末尾两位重复的情况)
    
    	while(sptr-- != shift)//该最外层循环完成给好后缀表中每一个单元进行赋值的工作
    	{
    		char *p1 = ptrn + pLen - 2, *p2,*p3;
    		
    		//该do...while循环完成以当前pptr所指的字符为边界时,要移动的距离
    		do{
    			while(p1 >= ptrn && *p1-- != c);//该空循环,寻找与最后一个字符c匹配的字符所指向的位置
    			
    			p2 = ptrn + pLen - 2;
    			p3 = p1;
    			
    			while(p3 >= ptrn && *p3-- == *p2-- && p2 >= pptr);//该空循环,判断在边界内字符匹配到了什么位置
    
    		}while(p3 >= ptrn && p2 >= pptr);
    
    		*sptr = shift + pLen - sptr + p2 - p3;//保存好后缀表中,以pptr所在字符为边界时,要移动的位置
    		/*
    		  PS:在这里我要声明一句,*sptr = (shift + pLen - sptr) + p2 - p3;
    		     大家看被我用括号括起来的部分,如果只需要计算字符串移动的距离,那么括号中的那部分是不需要的。
    			 因为在字符串自左向右做匹配的时候,指标是一直向左移的,这里*sptr保存的内容,实际是指标要移动
    			 距离,而不是字符串移动的距离。我想SNORT是出于性能上的考虑,才这么做的。			 
    		*/
    
    		pptr--;//边界继续向前移动
    	}
    
    	return shift;
    }
    
    
    /*
    	函数:int* BMSearch(char *, int , char *, int, int *, int *)
    	目的:判断文本串T中是否包含模式串P
    	参数:
    	    buf => 文本串T
    		blen => 文本串T长度
    		ptrn => 模式串P
    		PLen => 模式串P长度
    		skip => 坏字符表
    		shift => 好后缀表
        返回:
    	    int - 1表示成功(文本串包含模式串),0表示失败(文本串不包含模式串)。
    */
    int BMSearch(char *buf, int blen, char *ptrn, int plen, int *skip, int *shift)
    {
    	int b_idx = plen;  
    	if (plen == 0)
    		return 1;
    	while (b_idx <= blen)//计算字符串是否匹配到了尽头
    	{
    		int p_idx = plen, skip_stride, shift_stride;
    		while (buf[--b_idx] == ptrn[--p_idx])//开始匹配
    		{
     			if (b_idx < 0)
    				return 0;
    			if (p_idx == 0)
    			{     
    				return 1;
    			}
    		}
    		skip_stride = skip[(unsigned char)buf[b_idx]];//根据坏字符规则计算跳跃的距离
    		shift_stride = shift[p_idx];//根据好后缀规则计算跳跃的距离
    		b_idx += (skip_stride > shift_stride) ? skip_stride : shift_stride;//取大者
    	}
    	return 0;
    }
    
    
    
    
    
    ////////////////////////////////////////
    int main()
    {
        vector<int> b;
        //b = Process("ababaa");
       // for(size_t i=0;i<b.size();++i) cout<<b[i];
        KMP("abababa","ba");
        return 0;
    }
    

     参考我的上一篇博文中的matrix67大神和阮一博大神的讲解,BM算法好麻烦,直接拿过来了。。。

  • 相关阅读:
    1033 To Fill or Not to Fill (25分)(贪心)
    CentOS(五)--Linux系统的分区概念
    Linux安装Oracle 11G过程(测试未写完)
    【VMware虚拟化解决方案】设计和配置VMware vCenter 5.5
    CentOS(四)--Linux系统的启动级别
    CentOS(三)--初识linux的文件系统以及用户组等概念
    CentOS(二)--初识linux的一些常用命令
    CentOS(一)--CentOS6.4环境搭建
    Linux c/c++图片传输功能(中级版)
    remote uptime 服务器程序
  • 原文地址:https://www.cnblogs.com/shenbingyu/p/4852172.html
Copyright © 2011-2022 走看看