KMP算法

KMP算法是字符串模式匹配當(dāng)中最經(jīng)典的算法苹享,原來大二學(xué)數(shù)據(jù)結(jié)構(gòu)的有講,但是當(dāng)時(shí)只是記住了原理弛矛,但不知道代碼實(shí)現(xiàn)够吩,今天終于是完成了KMP的代碼實(shí)現(xiàn)。

原理

KMP的原理其實(shí)很簡(jiǎn)單丈氓,給定一個(gè)字符串和一個(gè)模式串周循,然后找模式串在給定字符串中的位置。將兩個(gè)字符串轉(zhuǎn)換為字符數(shù)組万俗,然后從兩個(gè)數(shù)組的開始位置"i"湾笛,"j"開始匹配,如果相同闰歪,執(zhí)行"i++"嚎研,"j++"接著比較下一位;如果不相同库倘,就轉(zhuǎn)到模式串對(duì)應(yīng)next數(shù)組的對(duì)應(yīng)位置"next[j]"然后從該位置開始繼續(xù)與給定字符串的當(dāng)前位置"i"進(jìn)行比較临扮,換句話說就是將模式串提前了"j-next[j]"位繼續(xù)比較,不至于每次出現(xiàn)不匹配就又重新回到開始位置進(jìn)行匹配于樟,充分利用了已匹配過的位置公条。

代碼

KMP算法的關(guān)鍵是得到模式串的next數(shù)組:
public static int[] next(char[] p) { int len = p.length; int[] next = new int[len]; next[0] = 0; next[1] = 0; //首先給next[0]和next[1]賦值拇囊,這兩個(gè)數(shù)字是固定的 for(int i = 2; i < len; i++) { int k = next[i - 1]; //用一個(gè)整型數(shù)字進(jìn)行遍歷 while(k >= 0) { if(p[i - 1] == p[k]) { next[i] = k + 1; //當(dāng)匹配到字符時(shí)就能得到當(dāng)前位置的next值迂曲,然后結(jié)束循環(huán) break; } k--; } } return next; }
得到next數(shù)組之后就可以進(jìn)行KMP匹配:
public int kmpSearch(char[] s, char[] p) { int i = 0, j = 0; //從0開始 int slen = s.length, plen = p.length; int[] next = next(p); while(i < slen && j < plen) { if(s[i] == p[j]) { //挨個(gè)進(jìn)行匹配 i++; j++; } else { j = next[j]; //如果不相等,返回next[j]位置繼續(xù)向后匹配寥袭,不用和前面的進(jìn)行比較 } } if(j == plen) //如果匹配到最后路捧,說明匹配成功,返回匹配成功的開始位置 return i - j; return -1; //否則就是匹配失敗传黄,返回-1 }
KMP算法還有一個(gè)進(jìn)階的next算法杰扫,求nextval數(shù)組:
public int[] nextVal(char[] p) { int len = p.length; int[] nextval = new int[len]; nextval[0] = -1; int i=-1, j = 0; while(j < len - 1) { if(i == -1 || p[j] == p[i]) { ++i; ++j; if(p[j] != p[i]) nextval[j] = i; else nextval[j] = nextval[i]; }else i = nextval[i]; } return nextval; }

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市膘掰,隨后出現(xiàn)的幾起案子章姓,更是在濱河造成了極大的恐慌佳遣,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,627評(píng)論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件凡伊,死亡現(xiàn)場(chǎng)離奇詭異零渐,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)系忙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,180評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門诵盼,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人银还,你說我怎么就攤上這事风宁。” “怎么了蛹疯?”我有些...
    開封第一講書人閱讀 169,346評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵戒财,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我捺弦,道長(zhǎng)固翰,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 60,097評(píng)論 1 300
  • 正文 為了忘掉前任羹呵,我火速辦了婚禮骂际,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘冈欢。我一直安慰自己歉铝,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,100評(píng)論 6 398
  • 文/花漫 我一把揭開白布凑耻。 她就那樣靜靜地躺著太示,像睡著了一般。 火紅的嫁衣襯著肌膚如雪香浩。 梳的紋絲不亂的頭發(fā)上类缤,一...
    開封第一講書人閱讀 52,696評(píng)論 1 312
  • 那天,我揣著相機(jī)與錄音邻吭,去河邊找鬼餐弱。 笑死,一個(gè)胖子當(dāng)著我的面吹牛囱晴,可吹牛的內(nèi)容都是我干的膏蚓。 我是一名探鬼主播,決...
    沈念sama閱讀 41,165評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼畸写,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼驮瞧!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起枯芬,我...
    開封第一講書人閱讀 40,108評(píng)論 0 277
  • 序言:老撾萬榮一對(duì)情侶失蹤论笔,失蹤者是張志新(化名)和其女友劉穎采郎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體狂魔,經(jīng)...
    沈念sama閱讀 46,646評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡尉剩,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,709評(píng)論 3 342
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了毅臊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片理茎。...
    茶點(diǎn)故事閱讀 40,861評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖管嬉,靈堂內(nèi)的尸體忽然破棺而出皂林,到底是詐尸還是另有隱情,我是刑警寧澤蚯撩,帶...
    沈念sama閱讀 36,527評(píng)論 5 351
  • 正文 年R本政府宣布础倍,位于F島的核電站,受9級(jí)特大地震影響胎挎,放射性物質(zhì)發(fā)生泄漏沟启。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,196評(píng)論 3 336
  • 文/蒙蒙 一犹菇、第九天 我趴在偏房一處隱蔽的房頂上張望德迹。 院中可真熱鬧,春花似錦揭芍、人聲如沸胳搞。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,698評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽肌毅。三九已至,卻和暖如春姑原,著一層夾襖步出監(jiān)牢的瞬間悬而,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,804評(píng)論 1 274
  • 我被黑心中介騙來泰國(guó)打工锭汛, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留笨奠,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 49,287評(píng)論 3 379
  • 正文 我出身青樓店乐,卻偏偏與公主長(zhǎng)得像艰躺,于是被迫代替她去往敵國(guó)和親呻袭。 傳聞我的和親對(duì)象是個(gè)殘疾皇子眨八,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,860評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容