青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

The Fourth Dimension Space

枯葉北風寒,忽然年以殘,念往昔,語默心酸。二十光陰無一物,韶光賤,寐難安; 不畏形影單,道途阻且慢,哪曲折,如渡飛湍。斬浪劈波酬壯志,同把酒,共言歡! -如夢令

深度剖析KMP,讓你認識真正的Next

KMP算法,想必大家都不陌生,它是求串匹配問題的一個經(jīng)典算法(當然如果你要理解成放電影的KMP,請退出本頁面直接登錄各大電影網(wǎng)站,謝謝),我想很多人對它的理解僅限于此,知道KMP能經(jīng)過預處理然后實現(xiàn)O(N*M)的效率,比brute force(暴力算法)更優(yōu)秀等等,其實KMP算法中的Next函數(shù),功能十分強大,其能力絕對不僅僅限于模式串匹配,它并不是KMP的附屬品,其實它還有更多不為人知的神秘功能^_^

先來看一個Next函數(shù)的典型應用,也就是模式串匹配,這個相信大家都很熟悉了:
POJ 3461 Oulipo——很典型的模式串匹配問題,求模式串在目標串中出現(xiàn)的次數(shù)。

#include<cstdio>
#include
<iostream>
#include
<cmath>
#include
<cstring>
#include
<algorithm>
using namespace std;
#define MAX 1000001

char t[MAX];
char s[MAX];
int next[MAX];
inline 
void calnext(char s[],int next[])
{
    
int i;
    
int j;
    
int len=strlen(s);
    next[
0]=-1;
    j
=-1;
    
for(i=1;i<len;i++)
    
{
        
while(j>=0&&s[i]!=s[j+1])
            j
=next[j];
        
if(s[j+1]==s[i])//上一個循環(huán)可能因為j=-1而不做,此時不能知道s[i]與s[j+1]的關系。故此需要此條件。
            j++;
        next[i]
=j;
    }

}



int KMP(char t[],char s[])
{
    
int ans=0;
    
int lent=strlen(t);
    
int lens=strlen(s);
    
if(lent<lens)
        
return 0;
    
int i,j;
    j
=-1;
    
for(i=0;i<lent;i++)
    
{

        
while(j>=0&&s[j+1]!=t[i])
            j
=next[j];
        
if(s[j+1]==t[i])
            j
++;
        
if(j==lens-1)
        
{
            ans
++;
            j
=next[j];
        }

    }

    
return ans;

}



int main()
{

    
int testcase;
    scanf(
"%d",&testcase);
    
int i;
    
for(i=1;i<=testcase;i++)
    
{
    
        scanf(
"%s%s",s,t);
        calnext(s,next);
        printf(
"%d\n",KMP(t,s));
    }

    
return 0;

}

——————————————————————————————————————————————————————————————————————————————————————
POJ 2406 Power Strings
這道題就比較有意思了,乍看之下,怎么看貌似都與KMP無關,呵呵,這就是因為你沒有深入理解Next 的含義;
我首先來解釋下這道題的題意,給你一個長度為n的字符串,首先我們找到這樣一個字符串,這個字符串滿足長度為n的字符串是由這個字符串重復疊加得到并且這個字符串的長度要最小.,然后輸出重復的次數(shù)。
比如說,ababab這個字符串,顯然它是由ab重復疊加3次得到,所以,答案輸出3.

那么這個題用next怎么做呢,我們必須知道,next數(shù)組里面存放的是 如果當前匹配失敗,模式串可以繼續(xù)進行匹配的下一個位置。
For Example:
        1 2 3 4 5 6
     S= a b a b a ?
  Next= 0 0 1 2 3
其中next[5]=3,說明如果模式串在j=6處匹配失敗,那么j=next[5]=3 ,為什么? 因為S的頭三位和末三位是一樣的,如果說S已經(jīng)匹配到5的位置(匹配到5說明前5位都已經(jīng)匹配上),那么前三位肯定也匹配上了(廢話~),若果這個時候要繼續(xù)匹配,我們可以將模式串向右平移幾個個單位,這樣保證S的前三位仍然是可以匹配上的。

比如說目標串是

    i=  1  2  3  4  5  6  7  8  9 
         a  b  a   b  a  d  e  f   g
         a  b  a   b  a  c
    j=  1  2  3  4  5  6
next= 0  0  1  2  3  ? 
現(xiàn)在我們發(fā)現(xiàn)i=6與j=6兩串不匹配怎么辦?由于next[5]指向3,那么我們將模式串平移
    i=  1  2  3  4  5  6  7  8  9 
         a  b  a   b  a  d  e  f   g
                 a   b  a  b  a  c   
           
j=  1  2  3  4  5  6
        next= 0  0  1  2  3  ? 
這樣我們從j=3處繼續(xù)向后匹配。(當然如果此時i=6與j=4仍然不匹配,那么我們繼續(xù)使用失敗函數(shù),去尋找下一個位置)

 好的,現(xiàn)在我們已經(jīng)知道了,next數(shù)組中所存放的數(shù)字的含義,那么接下來讓我們來看看怎樣靈活的使用next,揭開它不為人知的另一面吧。
回到原題,原題要求求出一個字符串的某一個子串,使得這個字符串不斷自我疊加后得到原串,并且這個重復的次數(shù)最多。那么它和next有什么關系呢???

結(jié)論:如果有一個字符串s,長度是len,它的失敗函數(shù)是next,如果len能被len-next[len]整除,那么len-next[len]就是我們要求的那個子串的長度,與之對應的字符串,就是我們想得到的子串;

為什么呢? 假設我們有一個字符串a(chǎn)babab,那么next[6]=4對吧,由于next的性質(zhì)是,匹配失敗后,下一個能繼續(xù)進行匹配的位置,也就是說,把字符串的前四個字母,abab,平移2個單位,這個abab一定與原串的abab重合(否則就不滿足失敗函數(shù)的性質(zhì)),這說明了什么呢,由于字符串進行了整體平移,而平移后又要重疊,那么必有
s[1]=s[3],s[2]=s[4],s[3]=s[5],s[4]=s[6].說明長度為2的字符串在原串中一定重復出現(xiàn),這就是len-next[len]的含義!
解決上面這個問題的同時,其實還有另一個問題,那就是如果這個字符串長度為奇數(shù)怎么辦?如ababa,好像next[len]也等于3呢,可是aba-ba-似乎并不是由ba重復得到的吧。我們先把這個字符串斷開,ab-ab-a,可以想象,中間的ab平移后,沒有ab與它重合(只能重合一個,這雖然沒有違背next的性質(zhì),但是卻對本題的方法造成了影響,請讀者細細品味),所以才會出現(xiàn)上面的情況!所以要加上len能夠整除len-next[len]這個條件.

此題源代碼如下:
//This is the source code for POJ 2406
//coded by abilitytao
//2009年7月31日11:17:34
#include<cstdio>
#include
<iostream>
#include
<cmath>
#include
<cstring>
#include
<algorithm>
using namespace std;
#define MAX 1000001

int next[MAX];
inline 
void calnext(char s[],int next[])
{
    
int i;
    
int j;
    
int len=strlen(s);
    next[
0]=-1;
    j
=-1;
    
for(i=1;i<len;i++)
    
{
        
while(j>=0&&s[i]!=s[j+1])
            j
=next[j];
        
if(s[j+1]==s[i])
            j++;
        next[i]
=j;
    }

}



int main()
{
    
int n;
    
char str[MAX];
    
int len;
    
while(scanf("%s",&str))
    
{
        
if(str[0]=='.')
            
break;
        len
=strlen(str);
        calnext(str,next);
        
if((len)%(len-1-next[len-1])==0)
            printf(
"%d\n",(len)/(len-1-next[len-1]));
        
else 
        
{
            putchar(
'1');
            putchar(
'\n');
        }


    }

    
return 0;
    
}



POJ 1961與上題類似,故不贅述,代碼如下:
//This is the source code for POJ 1961
//coded by abilitytao
//2009年7月31日10:56:07
#include<cstdio>
#include
<iostream>
#include
<cmath>
#include
<cstring>
#include
<algorithm>
using namespace std;
#define MAX 1000001

int next[MAX];
void calnext(char s[],int next[])
{
    
int i;
    
int j;
    
int len=strlen(s);
    next[
0]=-1;
    j
=-1;
    
for(i=1;i<len;i++)
    
{
        
while(j>=0&&s[i]!=s[j+1])
            j
=next[j];
        
if(s[j+1]==s[i])
            j
++;
        next[i]
=j;
    }

}



int main()
{
    
int n;
    
char str[MAX];
    
int casenum=0;
    
int i;
    
int len;
    
while(scanf("%d",&n))
    
{
        casenum
++;
        
        
if(n==0)
            
break;
        scanf(
"%s",str);
        len
=strlen(str);
        printf(
"Test case #%d\n",casenum);
        calnext(str,next);
        
for(i=1;i<len;i++)
        
{
            
            
if((i+1)%(i-next[i])==0&&next[i]!=-1)
                printf(
"%d %d\n",i+1,(i+1)/(i-next[i]));
        }

        printf(
"\n");
    }

    
return 0;
    
}




POJ 2752 Seek the Name, Seek the Fame
這道題揭開了next的另一個應用^_^
題目的意思可以這樣描述:給出一個字符串S,長度為len;找出一個前綴一個后綴,使得這兩個字符串相同。 輸出所有可能的情況。

如aaaaa,
aaaaa ——》OK
aaaaa ——》OK
aaaaa + aaaaa  ——》OK
aaaaa + aaaaa  ——》OK
aaaaa + aaaaa ——》OK

那么這個題怎么用next呢,其實很簡單,只要你知道next的含義。 s[1]——s[next[len]]中的內(nèi)容一定能與s[1+len-next[len]]——s[len]匹配,所以s[1]——s[next[len]]就是我們要求取的最長的那個串,然后呢我們循環(huán)地利用next,由于next的性質(zhì),可以保證,每一次得出的字串都能匹配到最后一個字母,也就是得到一個前綴等于后綴。只不過這個字符串的長度在不斷地減小罷了。不斷地使用next我們直到求出所有的前綴^_^  So the problem is cleared.

附源代碼:

//This is the source code for POJ 2752
//coded by abilitytao
//2009年7月31日12:17:45

#include
<cstdio>
#include
<iostream>
#include
<cmath>
#include
<cstring>
#include
<algorithm>
using namespace std;
#define MAX 1000001

int next[MAX];
inline 
void calnext(char s[],int next[])
{
    
int i;
    
int j;
    
int len=strlen(s);
    next[
0]=-1;
    j
=-1;
    
for(i=1;i<len;i++)
    
{
        
while(j>=0&&s[i]!=s[j+1])
            j
=next[j];
        
if(s[j+1]==s[i])//上一個循環(huán)可能因為j=-1而不做,此時不能知道s[i]與s[j+1]的關系。故此需要此條件。
            j++;
        next[i]
=j;
    }

}



int record[MAX];


int main()
{
    
char str[MAX];
    
int len;
    
int p=0;
    
int j;
    
while(scanf("%s",&str)!=EOF)
    
{
        calnext(str,next);
        p
=0;
        len
=strlen(str);
        j
=len-1;

        
while(j!=-1)
        
{
            record[
++p]=j;
            j
=next[j];
        }

        
while(p>1)
        

            printf(
"%d ",record[p]+1);
            p
--;
        }

            printf(
"%d\n",record[p]+1);
    }

    
return 0;
    
}


文章寫完了,如果有什么疏漏,還請大家批評指正~
文章來自abilitytao博客
轉(zhuǎn)載請注明出處:http://m.shnenglu.com/abilitytao/archive/2009/08/01/91865.html

posted on 2009-08-01 10:26 abilitytao 閱讀(3077) 評論(3)  編輯 收藏 引用

評論

# re: 深度剖析KMP,讓你認識真正的Next 2009-08-01 10:29 凡客誠品

不錯啊  回復  更多評論   

# re: 深度剖析KMP,讓你認識真正的Next 2009-08-01 16:09 Vincent

ac自動機,擴展kmp  回復  更多評論   

# re: 深度剖析KMP,讓你認識真正的Next 2009-08-01 16:56 abilitytao

@Vincent
呵呵 自動機絕對很強大的  回復  更多評論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            猫咪成人在线观看| 久久久久久久久久看片| 亚洲肉体裸体xxxx137| 久久精品国产99国产精品澳门| 亚洲欧洲精品一区二区三区| 久久精品视频va| 国产亚洲综合在线| 久久露脸国产精品| 久久久亚洲成人| 亚洲精品久久久一区二区三区| 欧美成人精品在线观看| 久久av一区| 亚洲韩国一区二区三区| 亚洲韩日在线| 国产精品美女久久久久久免费| 久久gogo国模裸体人体| 欧美一二三视频| 亚洲黑丝在线| 久久国产66| 久久久久久香蕉网| 一区二区三区欧美激情| 六月婷婷久久| 久久久久久久性| 亚洲精品激情| 亚洲国产精品一区二区久 | 欧美一区二区视频在线| 欧美—级a级欧美特级ar全黄| 国内久久婷婷综合| 99这里有精品| 亚洲精品欧美精品| 久久激情五月激情| 亚洲网站在线观看| 欧美精品www| 亚洲高清二区| 亚洲精品乱码久久久久久日本蜜臀| 亚洲一区二区三区四区在线观看 | 欧美激情一区三区| 国产日韩在线亚洲字幕中文| 99视频超级精品| 欧美日韩精品在线观看| 欧美国产先锋| 亚洲精品美女| 欧美精品七区| 99国产精品国产精品久久| 99re热这里只有精品视频| 牛牛影视久久网| 亚洲福利国产| 亚洲少妇中出一区| 欧美午夜精品久久久| 亚洲图片欧美日产| 欧美中文字幕在线| 国产日韩精品视频一区二区三区| 一区二区免费在线视频| 欧美一区二区视频观看视频| 国产欧美精品一区| 鲁大师成人一区二区三区 | 欧美一区二区三区免费看| 国产色婷婷国产综合在线理论片a| 香蕉久久夜色精品国产| 免费在线观看一区二区| 亚洲精品日产精品乱码不卡| 欧美激情精品久久久久久免费印度| 国产真实乱偷精品视频免| 你懂的国产精品永久在线| 一区二区三区免费在线观看| 欧美专区在线观看| 日韩午夜在线电影| 国产一区二区高清| 欧美日韩专区在线| 欧美激情一区二区三区蜜桃视频| 亚洲欧美日韩国产中文在线| 91久久在线观看| 男人的天堂亚洲| 欧美尤物巨大精品爽| 亚洲在线视频| 亚洲欧美国产三级| 亚洲小说欧美另类婷婷| 日韩小视频在线观看| 亚洲欧洲在线观看| 亚洲国产精品999| 精品成人久久| 在线日本欧美| 亚洲国产综合视频在线观看| 国产亚洲精品高潮| 国产精品一区二区你懂得| 男女精品网站| 久久夜色精品国产| 欧美在线免费| 一本一本久久a久久精品综合妖精 一本一本久久a久久精品综合麻豆 | 欧美大片在线观看一区二区| 午夜精品三级视频福利| 亚洲最新色图| 亚洲欧洲三级| 亚洲欧洲日产国产综合网| 久久久久久久久一区二区| 老司机免费视频一区二区| 亚洲欧美激情精品一区二区| 在线视频欧美日韩精品| 夜夜嗨av一区二区三区中文字幕 | 欧美电影免费网站| 欧美精品一区二区三区视频| 久久噜噜噜精品国产亚洲综合 | 麻豆国产精品777777在线| 亚洲国产成人在线| 国产精品一区二区你懂得 | 欧美sm视频| 欧美激情亚洲另类| 亚洲精品国产精品国自产在线 | 久久激情五月激情| 欧美伊人久久久久久午夜久久久久 | 欧美大片免费观看在线观看网站推荐| 久久国产主播| 亚洲国产精品视频| 亚洲欧美日韩视频二区| 亚洲一级黄色av| 欧美与黑人午夜性猛交久久久| 欧美一区国产一区| 欧美激情在线观看| 夜夜嗨av一区二区三区中文字幕 | 欧美一二区视频| 亚洲国产高清在线观看视频| 亚洲精品在线视频观看| 亚洲欧美日韩直播| 欧美二区在线观看| 国内精品一区二区| 午夜久久久久久久久久一区二区| 免费国产自线拍一欧美视频| 亚洲图片你懂的| 欧美日韩精品免费观看视频| 欧美黄色一区| 国产美女高潮久久白浆| 亚洲午夜91| 亚洲私人影院| 国产精品免费一区二区三区观看 | 欧美亚洲视频在线看网址| 午夜精彩视频在线观看不卡| 欧美激情一区二区三区蜜桃视频 | 羞羞漫画18久久大片| 欧美视频一区| 亚洲欧美综合v| 亚洲一区二区三区777| 国产精品久久九九| 欧美伊人久久大香线蕉综合69| 亚洲午夜国产成人av电影男同| 国产精品大全| 久久精品视频在线看| 欧美一区二区三区精品| 国内偷自视频区视频综合| 久久这里有精品15一区二区三区| 久久精品免费| 一区二区三区精品视频| 中文久久精品| 尤物yw午夜国产精品视频明星| 欧美成人午夜影院| 国产精品一区二区三区四区 | 午夜国产不卡在线观看视频| 在线视频你懂得一区| 欧美午夜电影一区| 久久人人爽人人爽爽久久| 免费看的黄色欧美网站| 午夜精品婷婷| 欧美成人自拍| 另类酷文…触手系列精品集v1小说| 麻豆91精品91久久久的内涵| 先锋影院在线亚洲| 欧美激情亚洲国产| 欧美福利一区二区| 国产色视频一区| 美女主播一区| 国产精品劲爆视频| 99综合电影在线视频| 亚洲激情午夜| 欧美成人激情在线| 久久青草久久| 永久久久久久| 久久青草久久| 老牛影视一区二区三区| 国产精品美女一区二区在线观看| 欧美国产日韩亚洲一区| 亚洲清纯自拍| 欧美aaa级| 亚洲靠逼com| 在线观看免费视频综合| 久久精品在线播放| 噜噜噜91成人网| 狠狠色综合网| 欧美mv日韩mv国产网站| 久热精品视频在线观看一区| 国产亚洲aⅴaaaaaa毛片| 欧美一级久久久| 欧美aa国产视频| 99视频在线精品国自产拍免费观看| 欧美sm重口味系列视频在线观看| 欧美亚洲免费| 一本色道综合亚洲| 欧美好骚综合网| 99re6这里只有精品视频在线观看| 欧美日本成人| 亚洲欧美三级在线| 欧美国产欧美亚洲国产日韩mv天天看完整 |