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

AC自動機算法詳解

    首先簡要介紹一下AC自動機:Aho-Corasick automation,該算法在1975年產生于貝爾實驗室,是著名的多模匹配算法之一。一個常見的例子就是給出n個單詞,再給出一段包含m個字符的文章,讓你找出有多少個單詞在文章里出現過。要搞懂AC自動機,先得有模式樹(字典樹)TrieKMP模式匹配算法的基礎知識。AC自動機算法分為3步:構造一棵Trie樹,構造失敗指針和模式匹配過程。
     如果你對KMP算法和了解的話,應該知道KMP算法中的next函數(shift函數或者fail函數)是干什么用的。KMP中我們用兩個指針ij分別表示,A[i-j+ 1..i]B[1..j]完全相等。也就是說,i是不斷增加的,隨著i的增加j相應地變化,且j滿足以A[i]結尾的長度為j的字符串正好匹配B串的前 j個字符,當A[i+1]B[j+1]KMP的策略是調整j的位置(減小j值)使得A[i-j+1..i]B[1..j]保持匹配且新的B[j+1]恰好與A[i+1]匹配,而next函數恰恰記錄了這個j應該調整到的位置。同樣AC自動機的失敗指針具有同樣的功能,也就是說當我們的模式串在Tire上進行匹配時,如果與當前節點的關鍵字不能繼續匹配的時候,就應該去當前節點的失敗指針所指向的節點繼續進行匹配。
      看下面這個例子:給定5個單詞:say she shr he her,然后給定一個字符串yasherhs問一共有多少單詞在這個字符串中出現過。我們先規定一下AC自動機所需要的一些數據結構,方便接下去的編程。
 1 const int kind = 26
 2 struct node{  
 3     node *fail;       //失敗指針
 4     node *next[kind]; //Tire每個節點的個子節點(最多個字母)
 5     int count;        //是否為該單詞的最后一個節點
 6     node(){           //構造函數初始化
 7         fail=NULL; 
 8         count=0
 9         memset(next,NULL,sizeof(next)); 
10     } 
11 }*q[500001];          //隊列,方便用于bfs構造失敗指針
12 char keyword[51];     //輸入的單詞
13 char str[1000001];    //模式串
14 int head,tail;        //隊列的頭尾指針

有了這些數據結構之后,就可以開始編程了:
   
首先,將這5個單詞構造成一棵Tire,如圖-1所示。

 

 1 void insert(char *str,node *root){ 
 2     node *p=root; 
 3     int i=0,index;  
 4     while(str[i]){ 
 5         index=str[i]-'a'
 6         if(p->next[index]==NULL) p->next[index]=new node();  
 7         p=p->next[index];
 8         i++;
 9     } 
10     p->count++;     //在單詞的最后一個節點count+1,代表一個單詞
11 }

在構造完這棵Tire之后,接下去的工作就是構造下失敗指針。構造失敗指針的過程概括起來就一句話:設這個節點上的字母為C,沿著他父親的失敗指針走,直到走到一個節點,他的兒子中也有字母為C的節點。然后把當前節點的失敗指針指向那個字母也為C的兒子。如果一直走到了root都沒找到,那就把失敗指針指向root。具體操作起來只需要:先把root加入隊列(root的失敗指針指向自己或者NULL),這以后我們每處理一個點,就把它的所有兒子加入隊列,隊列為空。

 1 void build_ac_automation(node *root){
 2     int i;
 3     root->fail=NULL; 
 4     q[head++]=root; 
 5     while(head!=tail){ 
 6         node *temp=q[tail++]; 
 7         node *p=NULL; 
 8         for(i=0;i<26;i++){ 
 9             if(temp->next[i]!=NULL){ 
10                 if(temp==root) temp->next[i]->fail=root;                 
11                 else
12                     p=temp->fail; 
13                     while(p!=NULL){  
14                         if(p->next[i]!=NULL){ 
15                             temp->next[i]->fail=p->next[i]; 
16                             break
17                         } 
18                         p=p->fail; 
19                     } 
20                     if(p==NULL) temp->next[i]->fail=root; 
21                 } 
22                 q[head++]=temp->next[i];  
23             } 
24         }   
25     } 
26 }

    從代碼觀察下構造失敗指針的流程:對照圖-2來看,首先rootfail指針指向NULL,然后root入隊,進入循環。第1次循環的時候,我們需要處理2個節點:root->next[‘h’-‘a’](節點h) root->next[‘s’-‘a’](節點s)。把這2個節點的失敗指針指向root,并且先后進入隊列,失敗指針的指向對應圖-2中的(1)(2)兩條虛線;第2次進入循環后,從隊列中先彈出h,接下來p指向h節點的fail指針指向的節點,也就是root;進入第13行的循環后,p=p->fail也就是p=NULL,這時退出循環,并把節點efail指針指向root,對應圖-2中的(3),然后節點e進入隊列;第3次循環時,彈出的第一個節點a的操作與上一步操作的節點e相同,把afail指針指向root,對應圖-2中的(4),并入隊;第4次進入循環時,彈出節點h(圖中左邊那個),這時操作略有不同。在程序運行到14行時,由于p->next[i]!=NULL(rooth這個兒子節點,圖中右邊那個),這樣便把左邊那個h節點的失敗指針指向右邊那個root的兒子節點h,對應圖-2中的(5),然后h入隊。以此類推:在循環結束后,所有的失敗指針就是圖-2中的這種形式。

  最后,我們便可以在AC自動機上查找模式串中出現過哪些單詞了。匹配過程分兩種情況:(1)當前字符匹配,表示從當前節點沿著樹邊有一條路徑可以到達目標字符,此時只需沿該路徑走向下一個節點繼續匹配即可,目標字符串指針移向下個字符繼續匹配;(2)當前字符不匹配,則去當前節點失敗指針所指向的字符繼續匹配,匹配過程隨著指針指向root結束。重復這2個過程中的任意一個,直到模式串走到結尾為止。
 1 int query(node *root){ 
 2     int i=0,cnt=0,index,len=strlen(str); 
 3     node *p=root;  
 4     while(str[i]){  
 5         index=str[i]-'a';  
 6         while(p->next[index]==NULL && p!=root) p=p->fail; 
 7         p=p->next[index]; 
 8         p=(p==NULL)?root:p; 
 9         node *temp=p; 
10         while(temp!=root && temp->count!=-1){ 
11             cnt+=temp->count; 
12             temp->count=-1
13             temp=temp->fail; 
14         } 
15         i++;                 
16     }    
17     return cnt; 
18 }
    對照圖-2,看一下模式匹配這個詳細的流程,其中模式串為yasherhs。對于i=0,1。Trie中沒有對應的路徑,故不做任何操作;i=2,3,4時,指針p走到左下節點e。因為節點e的count信息為1,所以cnt+1,并且講節點e的count值設置為-1,表示改單詞已經出現過了,防止重復計數,最后temp指向e節點的失敗指針所指向的節點繼續查找,以此類推,最后temp指向root,退出while循環,這個過程中count增加了2。表示找到了2個單詞she和he。當i=5時,程序進入第5行,p指向其失敗指針的節點,也就是右邊那個e節點,隨后在第6行指向r節點,r節點的count值為1,從而count+1,循環直到temp指向root為止。最后i=6,7時,找不到任何匹配,匹配過程結束。

    到此為止AC自動機算法的詳細過程已經全部介紹結束,看一道例題:http://acm.hdu.edu.cn/showproblem.php?pid=2222
Problem Description
In the modern time, Search engine came into the life of everybody like Google, Baidu, etc.
Wiskey also wants to bring this feature to his image retrieval system.
Every image have a long description, when users type some keywords to find the image, the system will match the keywords with description of image and show the image which the most keywords be matched.
To simplify the problem, giving you a description of image, and some keywords, you should tell me how many keywords will be match.
 

Input
First line will contain one integer means how many cases will follow by.
Each case will contain two integers N means the number of keywords and N keywords follow. (N <= 10000)
Each keyword will only contains characters 'a'-'z', and the length will be not longer than 50.
The last line is the description, and the length will be not longer than 1000000.
 

Output
Print how many keywords are contained in the description.
 

Sample Input
1
5
she
he
say
shr
her
yasherhs
 

Sample Output
3

 1 #include <iostream> 
 2 using namespace std; 
 3   
 4 const int kind = 26
 5 struct node{  
 6     node *fail;       //失敗指針
 7     node *next[kind]; //Tire每個節點的26個子節點(最多26個字母)
 8     int count;        //是否為該單詞的最后一個節點
 9     node(){           //構造函數初始化
10         fail=NULL; 
11         count=0
12         memset(next,NULL,sizeof(next)); 
13     } 
14 }*q[500001];          //隊列,方便用于bfs構造失敗指針
15 char keyword[51];     //輸入的單詞
16 char str[1000001];    //模式串
17 int head,tail;        //隊列的頭尾指針
18   
19 void insert(char *str,node *root){ 
20     node *p=root; 
21     int i=0,index;  
22     while(str[i]){ 
23         index=str[i]-'a'
24         if(p->next[index]==NULL) p->next[index]=new node();  
25         p=p->next[index];
26         i++;
27     } 
28     p->count++
29 
30 void build_ac_automation(node *root){
31     int i;
32     root->fail=NULL; 
33     q[head++]=root; 
34     while(head!=tail){ 
35         node *temp=q[tail++]; 
36         node *p=NULL; 
37         for(i=0;i<26;i++){ 
38             if(temp->next[i]!=NULL){ 
39                 if(temp==root) temp->next[i]->fail=root;                 
40                 else
41                     p=temp->fail; 
42                     while(p!=NULL){  
43                         if(p->next[i]!=NULL){ 
44                             temp->next[i]->fail=p->next[i]; 
45                             break
46                         } 
47                         p=p->fail; 
48                     } 
49                     if(p==NULL) temp->next[i]->fail=root; 
50                 } 
51                 q[head++]=temp->next[i];  
52             } 
53         }   
54     } 
55 
56 int query(node *root){ 
57     int i=0,cnt=0,index,len=strlen(str); 
58     node *p=root;  
59     while(str[i]){  
60         index=str[i]-'a';  
61         while(p->next[index]==NULL && p!=root) p=p->fail; 
62         p=p->next[index]; 
63         p=(p==NULL)?root:p; 
64         node *temp=p; 
65         while(temp!=root && temp->count!=-1){ 
66             cnt+=temp->count; 
67             temp->count=-1
68             temp=temp->fail; 
69         } 
70         i++;                 
71     }    
72     return cnt; 
73 
74 int main(){ 
75     int n,t; 
76     scanf("%d",&t); 
77     while(t--){  
78         head=tail=0
79         node *root=new node(); 
80         scanf("%d",&n); 
81         getchar(); 
82         while(n--){ 
83             gets(keyword); 
84             insert(keyword,root); 
85         } 
86         build_ac_automation(root); 
87         scanf("%s",str); 
88         printf("%d\n",query(root));  
89     } 
90     return 0
91 }

    PS:原創,轉載請注明出處

posted on 2009-04-21 18:22 極限定律 閱讀(68567) 評論(40)  編輯 收藏 引用 所屬分類: ACM/ICPC

評論

# re: AC自動機算法詳解 2009-06-13 09:53 zab08

bfs寫成 頭入尾出...

  回復  更多評論   

# re: AC自動機算法詳解 2009-07-30 15:39 xtu715

您好,對于例題,請測試如下樣例:
1
2
tacab
aca
wqzpacakkk

您程序給出的答案是1.
可根據題意,正確答案應該是2.

您代碼中65行,貌似有誤。

  回復  更多評論   

# re: AC自動機算法詳解 2009-07-30 15:49 xtu715

是我的錯,請無視。不好意思  回復  更多評論   

# re: AC自動機算法詳解 2009-08-25 14:00 zeus

沒想到又到你這里來 呵呵 看看先  回復  更多評論   

# re: AC自動機算法詳解 2009-08-26 15:47 路過

1
5
bhea
her
he
h
ha
bhera
輸出?  回復  更多評論   

# re: AC自動機算法詳解[未登錄] 2009-10-16 11:35 DD

65行的代碼temp->count!=-1這個條件不能要,否則就是錯誤的。  回復  更多評論   

# re: AC自動機算法詳解 2010-01-11 21:46 niubi

是主串在trie上匹配,不是模式串吧。。  回復  更多評論   

# re: AC自動機算法詳解[未登錄] 2010-02-26 13:09 intheway

好東東,樓主辛苦了.  回復  更多評論   

# re: AC自動機算法詳解 2010-07-14 00:38 MJ_LC

@DD
如果temp->count!=-1說明該節點和之后的節點已經計算過了  回復  更多評論   

# re: AC自動機算法詳解 2010-12-07 22:32 周燎明

@MJ_LC

請考慮第66與67 行,考慮將67行加一個執行判斷條件:if count==1 再做賦值操作。。。  回復  更多評論   

# re: AC自動機算法詳解[未登錄] 2010-12-22 20:03 AC

失敗指針構造的描述好模糊。看了半天還是沒看懂。。繼續再看  回復  更多評論   

# re: AC自動機算法詳解 2011-03-10 21:44 dereky

膜拜了  回復  更多評論   

# re: AC自動機算法詳解 2011-04-08 11:19

你的代碼錯了 失敗函數寫得有問題  回復  更多評論   

# re: AC自動機算法詳解 2011-05-28 19:45 anmtcel

您的程序會怎么處理模式串為
abcd和bc這兩個,
主串為abcde的這個問題?
應該是兩個串都出現了
但您的程序應該只會找到第一個串,

因為從a一直順利的走向d,但d的失敗指針直接指向根了,
也就是不會走到bc那條小路里去。

我有個想法,就是需不需要預處理出哪些串是被當前串包含的,
存當前串所包含的串的位置,也就是當當前串被查找出來后,這些位置也標記為已查找  回復  更多評論   

# re: AC自動機算法詳解 2011-06-19 20:15 fotile

我只想說,您的代碼實在不漂亮。  回復  更多評論   

# re: AC自動機算法詳解 2011-06-19 20:23 fotile

講解很淺顯易懂,代碼里一串一串的!=NULL還要很多不必要的判斷實在是配不上文字的高超。  回復  更多評論   

# re: AC自動機算法詳解 2011-07-03 13:46 酷行天下

大牛講的非常透徹,贊一個……  回復  更多評論   

# re: AC自動機算法詳解 2011-07-31 21:23 aa

膜拜  回復  更多評論   

# re: AC自動機算法詳解 2011-08-13 14:24 aaa

各種trie寫成tire  回復  更多評論   

# re: AC自動機算法詳解 2011-08-19 22:15 Evan

@路過
應該是4 吧。  回復  更多評論   

# re: AC自動機算法詳解 2011-10-21 11:04 mmx

好!  回復  更多評論   

# re: AC自動機算法詳解 2011-12-01 01:58 Miu_Y

節點申請了不釋放?  回復  更多評論   

# re: AC自動機算法詳解 2012-02-27 20:55 ITX351

57行為什么要求長度= = 不理解  回復  更多評論   

# re: AC自動機算法詳解 2012-04-09 17:18 himdd

char str[1000001];應該叫目標串吧。。  回復  更多評論   

# re: AC自動機算法詳解 2012-04-26 09:57 Lucian

內容很棒,很容易理解!  回復  更多評論   

# re: AC自動機算法詳解 2012-05-16 11:25 wzq

多組數據的話,每組做完指針是不是應該釋放呢?  回復  更多評論   

# re: AC自動機算法詳解 2012-07-15 23:12 re

代碼bug超多  回復  更多評論   

# re: AC自動機算法詳解 2012-09-13 12:06 cqwelly

頂一下  回復  更多評論   

# re: AC自動機算法詳解 2012-12-24 18:22 陳文康

@xtu715
為啥是2??  回復  更多評論   

# re: AC自動機算法詳解 2013-02-17 16:42 h

temp->count!=-1
注意:1、重復的只算一次;2、初始的count=0
模式串為
abcd和bc這兩個,
主串為abcde的這個問題?
當找到c時,count[c]=0,tmp指針找到串bc的結尾c并統計。
  回復  更多評論   

# re: AC自動機算法詳解 2013-05-02 21:11 xero essential

其實程序上的錯誤無傷大雅,個人感覺看了第一段就明白原理了,剩下的只是驗證。。  回復  更多評論   

# re: AC自動機算法詳解[未登錄] 2013-08-10 21:47 111

好東西,謝了!  回復  更多評論   

# re: AC自動機算法詳解 2014-03-09 10:03 testcc

不會內存泄露么??  回復  更多評論   

# re: AC自動機算法詳解 2014-03-25 10:28 Islo

orz  回復  更多評論   

# re: AC自動機算法詳解 2014-07-29 16:28 lb

先把root加入隊列(root的失敗指針指向自己或者NULL),這以后我們每處理一個點,就把它的所有兒子加入隊列,隊列為空。
這句話什么意思  回復  更多評論   

# re: AC自動機算法詳解 2014-09-12 12:55 Vmurder

@xtu715
為什么是2?  回復  更多評論   

# re: AC自動機算法詳解 2015-05-10 11:23

http://sunmoon-template.blogspot.tw/2015/05/aho-corasick-automation-ac.html
構造的地方可以更簡潔  回復  更多評論   

# re: AC自動機算法詳解 2015-07-26 19:13 @anyi

真好  回復  更多評論   

# re: AC自動機算法詳解 2015-08-20 11:30 fastmatch

@anmtcel
abcd和bc不在一個分支上  回復  更多評論   

<2009年6月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

導航

統計

常用鏈接

留言簿(10)

隨筆分類

隨筆檔案

友情鏈接

搜索

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            蜜桃久久av一区| 亚洲国产精品va在线看黑人 | 在线不卡a资源高清| 久久久人人人| 欧美一区二区播放| 久久久国产午夜精品| 国产精品久久久一本精品| 欧美一区二区在线免费播放| 亚洲尤物视频在线| 日韩西西人体444www| 久久av老司机精品网站导航| 99国产一区| 中文欧美日韩| 久久久国产精品一区二区中文 | 一区二区三区欧美在线观看| 亚洲亚洲精品三区日韩精品在线视频 | 欧美久久久久久久久久| 亚洲国产欧美久久| 亚洲人成网站影音先锋播放| 亚洲电影在线观看| 亚洲精品一区二区三区婷婷月| 久久精品在线观看| 亚洲一区二区免费视频| 久久精品成人一区二区三区蜜臀| 久久亚洲综合色| 欧美午夜在线观看| 狠狠色综合日日| 一区二区三区四区精品| 久久久久国产精品一区| 亚洲激情视频在线观看| 午夜精品一区二区三区四区| 蜜月aⅴ免费一区二区三区| 国产精品久久777777毛茸茸| 亚洲第一区在线| 久久精品欧洲| 在线综合+亚洲+欧美中文字幕| 蜜桃久久av| 国产综合网站| 欧美亚洲日本一区| 一本色道久久综合亚洲91| 免费成人美女女| 好吊色欧美一区二区三区四区| 一区二区三区视频免费在线观看| 一本久道综合久久精品| 国产一区二区中文| 亚洲综合视频网| 91久久精品一区二区别| 久久久之久亚州精品露出| 国产精品国产三级国产专区53| 亚洲精品网站在线播放gif| 老司机成人网| 久久精品国产亚洲aⅴ| 国产午夜精品久久久久久久| 亚洲欧美视频一区二区三区| 亚洲老司机av| 欧美日韩精品欧美日韩精品| 亚洲精品欧美日韩专区| 欧美成黄导航| 麻豆精品精品国产自在97香蕉| 国产在线精品一区二区夜色| 久久精品九九| 欧美在线观看视频一区二区三区| 国产女人18毛片水18精品| 性伦欧美刺激片在线观看| 亚洲午夜日本在线观看| 国产精品一区二区久久久| 亚洲自拍偷拍视频| 国产精品黄色在线观看| 香蕉久久一区二区不卡无毒影院| 亚洲免费成人av| 欧美福利电影网| 欧美在线观看视频一区二区| 美女脱光内衣内裤视频久久影院 | 欧美日韩国产色视频| 亚洲国产综合在线| 亚洲国产视频一区二区| 欧美激情亚洲综合一区| 一区二区三区高清视频在线观看 | 免费观看日韩av| 久久亚洲综合色| 日韩天堂在线视频| 亚洲桃花岛网站| 激情丁香综合| 亚洲福利视频一区二区| 欧美日本亚洲| 久久av一区二区三区漫画| 久久精品国产综合精品| 亚洲欧洲精品一区二区三区波多野1战4 | 亚洲男人av电影| 国产精品免费区二区三区观看| 亚洲一区在线直播| 欧美伊人久久久久久久久影院| 在线观看欧美成人| 一片黄亚洲嫩模| 黄色成人av网| 一区二区日韩免费看| 黄色一区二区在线| 99视频精品免费观看| 红桃视频一区| 亚洲淫性视频| 一本色道久久88综合亚洲精品ⅰ | 一本色道精品久久一区二区三区 | 国产精品v日韩精品| 久久久爽爽爽美女图片| 欧美视频在线一区二区三区| 米奇777超碰欧美日韩亚洲| 欧美色中文字幕| 免费成人在线视频网站| 国产精品嫩草影院一区二区| 欧美国产另类| 一区二区三区不卡视频在线观看| 亚洲电影自拍| 欧美日韩一区二区三区在线观看免 | 一区二区三区成人| 亚洲电影在线播放| 久热精品在线视频| 国产性色一区二区| 欧美成人中文| 国产色爱av资源综合区| 99国产一区| av成人免费| 欧美a级片一区| 欧美亚洲自偷自偷| 欧美日韩亚洲一区二区| 欧美主播一区二区三区美女 久久精品人 | 欧美在线免费视屏| 欧美成人精品不卡视频在线观看 | 久久激情五月丁香伊人| 亚洲性感激情| 欧美日韩国产在线一区| 亚洲欧洲偷拍精品| 亚洲激情专区| 老牛嫩草一区二区三区日本| 久久一二三四| 在线看国产一区| 久久尤物视频| 欧美本精品男人aⅴ天堂| 一区免费在线| 久久琪琪电影院| 欧美不卡视频一区发布| 午夜亚洲伦理| 亚洲网站在线播放| 欧美日韩另类丝袜其他| 亚洲精品视频免费在线观看| 日韩视频一区二区在线观看| 欧美gay视频| 亚洲美女视频在线免费观看| 一级日韩一区在线观看| 欧美日韩亚洲高清一区二区| 日韩一级黄色av| 亚洲欧美日韩综合aⅴ视频| 国产精品二区二区三区| 亚洲欧美电影在线观看| 久久久久.com| 亚洲激情小视频| 欧美体内谢she精2性欧美| 午夜精品久久久久久久久久久| 亚洲美女淫视频| 亚洲高清123| 亚洲欧洲日本mm| 亚洲免费伊人电影在线观看av| 久久精品在这里| 欧美激情黄色片| 亚洲美女诱惑| 欧美性色aⅴ视频一区日韩精品| 亚洲精选一区| 欧美在线视频免费播放| 亚洲大片精品永久免费| 欧美日韩一区二区视频在线观看| 亚洲性视频h| 免费的成人av| 国产精品乱看| 久久精品国产精品亚洲精品| 美女日韩欧美| 正在播放亚洲| 国产一区二区三区在线观看视频| 久久亚洲风情| 亚洲视频一区二区免费在线观看| 久久久久国产免费免费| 中文精品视频| 激情伊人五月天久久综合| 欧美日本在线看| 欧美一区二区视频在线观看| 最新国产の精品合集bt伙计| 久久国内精品自在自线400部| 最新69国产成人精品视频免费| 国产精品日韩精品欧美在线| 欧美大片在线观看一区二区| 午夜免费久久久久| 99精品欧美| 亚洲电影免费| 老司机久久99久久精品播放免费 | 最近看过的日韩成人| 国产精品揄拍一区二区| 欧美精品1区2区| 久久久在线视频| 午夜精品偷拍| 亚洲自拍偷拍视频| 一区二区三区导航| 最新精品在线|