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

AC自動機

Posted on 2011-10-19 19:03 Mato_No1 閱讀(607) 評論(0)  編輯 收藏 引用
AC自動機就是在Trie樹上加入一些失敗指針(fail,類似KMP中的next),使得它在某個結點失配的時候能夠轉移到該結點失敗指針指向的結點繼續匹配,從而實現多串匹配(單主串多子串)。

【1】Trie樹的結構:
首先是結點類型:
struct node {
    
int mul, ch[SZ], fail;
} T[MAXN];
其中SZ是字符集的大小,比如小寫字母集SZ=26,數字集SZ=10等。
另外這個mul表示的是該結點的重復次數(和平衡樹中的比較像),就是這個結點所對應的字符串(從根到該結點路徑上的所有邊上的字符組成的字符串)出現了幾次。
(不過這種表示法MS不是很好……如果有卡空間的,比如結點總數和SZ之積達到了100M以上,而真正的邊又很少的時候……就囧掉了……而如果用指針的話在Linux下面又會CE掉……各位神犇來說一下怎么解決啊囧……)
然后,Trie樹的0號結點(T[0])是空結點(用來代替指針中的NULL),因此真正結點下標從1開始。1號結點(T[1])一般都作為根結點,所以下面直接寫#define root 1。
還有(這句話是除草用的……)Trie樹的字符全部都在邊上而不在點上,因此T[x].ch[c]其實指的是“結點x引出的字符為c的邊所指向的結點”,簡稱結點x的c子結點。

【2】自動機的建立:
首先要建立Trie樹(也就是在空的Trie樹上插入所有要匹配的子串)。這個隨便搞一下就傻掉了,因此直接上代碼:
void ins()
{
    
int len = s0.length(), x = root, c;
    re(i, len) {
        c 
= s0[i] - 97;
        
if (!T[x].ch[c]) {T[x].ch[c] = ++N; T[N].mul = 0; re(j, SZ) T[N].ch[j] = 0;}
        x 
= T[x].ch[c];
    }
    T[x].mul
++;
}
這里string s0是待插入的字符串,定義成全局變量,因為在參數中出現string有可能爆掉。

然后就是建立自動機了。
這一過程其實是用BFS遍歷來實現的。首先,T[root].fail=0(root也是整棵樹中唯一的fail為0的結點)并將root入隊。下面按照BFS的順序依次取出隊所有的點,對于結點i,若它存在k子結點j(也就是j=T[i].ch[k],且j≠0),則結點j入隊,并計算j的失敗指針fail,方法:從T[i].fail(不是i)開始不斷上溯,直到找到一個存在k子結點的結點或者到root都木有找到(結點下標為0,即NULL)。若找到了一個存在k子結點的結點x,則將T[j].fail置為T[x].ch[k],否則(直到root都木有找到),T[j].fail=root。

到這里失敗指針的用處就顯現了:如果匹配到結點x的時候失配(即接下來的一個字符是c而T[x].ch[c]=0),可以立刻轉到T[x].fail進行匹配,因為T[x].fail有以下三個特征:
<1>其深度嚴格小于x的深度;
<2>其代表的字符串一定是x代表的字符串的后綴;
<3>該結點一定是滿足條件<1><2>的所有結點中深度最小的結點;

代碼:
void mkf()
{
    Q[
0= root; T[root].fail = 0;
    
int i, j, x;
    
for (int front=0, rear=0; front<=rear; front++) {
        i 
= Q[front];
        re(k, SZ) 
if (j = T[i].ch[k]) {
            x 
= T[i].fail;
            
while (x && !T[x].ch[k]) x = T[x].fail;
            
if (x) T[j].fail = T[x].ch[k]; else T[j].fail = root;
            Q[
++rear] = j;
        }
    }
}

【3】匹配過程:
在有了失敗指針時,其匹配過程就和KMP差不多了。
設主串為A(代碼中同),依次掃描A[0..A.length()-1]進行匹配。設目前匹配到了第i位,A[i]=c,當前結點為x。匹配過程中將進行以下操作:
<1>成功匹配(T[x]有c子結點),則進入T[x]的c子結點;
<2>失配(T[x]無c子結點),則從T[x].fail開始,沿著失敗指針上溯,直到找到一個有c子結點的結點為止。如果到了root都木有找到這樣的結點,則停止在root處;
<3>設立結點y,從當前的結點x開始不斷上溯到root(注意root也要算,因為子串中可能有空串),將中間所有結點的mul值累加進最終結果(表示這些字符串在主串中出現了,統計次數),如果題目中要求統計不重復的子串個數(如HDU2222),則在累加之后將它們全部置為0,防止下次再次累加。這一步操作實質上就是統計A中所有以A[i]為右端點的子串個數。

這樣,整個過程就傻掉了。

代碼:
void solve()
{
    
int len = A.length(), x = root, y, c; res = 0;
    re(i, len) {
        c 
= A[i] - 97;
        
while (x && !T[x].ch[c]) x = T[x].fail;
        
if (!x) x = root; else x = T[x].ch[c];
        y 
= x;
        
while (y) {res += T[y].mul; T[y].mul = 0; y = T[y].fail;}
    }
}

有關例題:
【1】HDU2222
裸的多串匹配問題,模板題;
有關該題的詳細資料(包括易疵點)見這里
【2】HDU2896
比2222稍難一些,但還是模板題。注意這題的子串木有重復的,因此mul可以設為bool。
【3】HDU3065
統計每個子串出現的次數(可以重復),也是模板題。
以上三題均不卡空間。
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美日韩第一区日日骚| 午夜视频一区二区| 欧美jjzz| 亚洲国语精品自产拍在线观看| 99在线精品观看| 亚洲男女毛片无遮挡| 国产精品美女久久久久aⅴ国产馆| 性色一区二区| 久久九九99| 亚洲一区在线播放| 亚洲高清电影| 欧美一区二区三区视频在线| 亚洲激情成人在线| 国产九九精品| 欧美日韩一级片在线观看| 久久精品视频99| 美女任你摸久久| 欧美一区二区三区四区在线观看地址 | 免费在线观看成人av| 欧美日本在线| 久久久久久久999精品视频| 亚洲专区一区二区三区| 欧美系列精品| 久久精品中文| 欧美国产精品一区| 久久亚洲影院| 久久爱另类一区二区小说| 亚洲欧美乱综合| 久久香蕉国产线看观看av| 欧美视频精品在线| 欧美久久久久| 黑人巨大精品欧美黑白配亚洲 | 久久久久久久性| 亚洲一区二区av电影| 亚洲精品乱码久久久久久久久| 国产亚洲欧美色| 国产视频在线一区二区 | 亚洲国产一区二区精品专区| 99视频精品在线| 亚洲精品视频一区二区三区| 欧美亚洲一区二区在线| 亚洲综合成人在线| 午夜精品理论片| 香港久久久电影| 亚洲国产你懂的| 久久久久国产精品人| 国产精品久久久久三级| 亚洲美女啪啪| 中文精品一区二区三区| 亚洲男人第一网站| 亚洲激情网址| 中日韩高清电影网| 亚洲在线播放电影| 欧美日韩高清在线| 亚洲美女电影在线| 亚洲大胆人体在线| 亚洲激情在线观看| 久久久久久久欧美精品| 国产在线欧美日韩| 日韩午夜在线播放| 亚洲欧美一区二区视频| 日韩视频不卡中文| 午夜精品亚洲| 国产欧美精品一区aⅴ影院| 黄色亚洲精品| 久久综合伊人77777尤物| 国产精品国产一区二区| 国产在线精品一区二区夜色| 性色一区二区| 午夜免费久久久久| 国产一区再线| 欧美jizzhd精品欧美巨大免费| 亚洲精品美女久久7777777| 亚洲欧美日韩一区二区在线 | 性xx色xx综合久久久xx| 午夜精品美女久久久久av福利| 国产精品人人爽人人做我的可爱| 国外精品视频| 免费久久99精品国产| 亚洲男人av电影| 国产欧美在线| 欧美.www| 欧美日韩国产一区二区三区| 亚洲综合视频在线| 久久av最新网址| 国产精品推荐精品| 国内偷自视频区视频综合| 久久精品色图| 亚洲一区二区伦理| 国产裸体写真av一区二区| 久久一区二区三区超碰国产精品| 亚洲一区综合| 亚洲成色777777女色窝| 久久久久久噜噜噜久久久精品| av不卡免费看| 亚洲精品久久7777| 国产午夜一区二区三区| 亚洲国产精品成人一区二区| 国产精品国产三级国产专播精品人| 久久精品国产第一区二区三区最新章节 | 亚洲国内精品| 国产精品色婷婷久久58| 美女爽到呻吟久久久久| 欧美午夜在线视频| 你懂的亚洲视频| 国产伦精品一区| 亚洲精品一二三| 老司机久久99久久精品播放免费| 国产女人aaa级久久久级| 欧美成人精品一区| 国产精品美女www爽爽爽视频| 欧美韩国在线| 欧美高清在线播放| 欧美在线观看视频| 欧美日韩另类字幕中文| 美女视频一区免费观看| 国产欧美精品日韩精品| 日韩午夜黄色| 亚洲欧洲精品天堂一级| 亚洲精品免费一二三区| 韩国久久久久| 亚洲欧美综合另类中字| 亚洲午夜精品久久久久久app| 美女精品在线观看| 久久久久久综合| 国产日韩欧美电影在线观看| 一本久久综合| 国产一区视频在线看| 一区二区三区久久久| av成人免费| 农村妇女精品| 欧美高清hd18日本| 欧美激情一区二区三区不卡| 一区二区三区www| 中文一区二区在线观看| 日韩视频一区二区三区在线播放免费观看 | 亚洲欧美日本国产有色| 欧美激情在线有限公司| 欧美成人激情视频| 亚洲国产高清高潮精品美女| 日韩午夜在线电影| 99成人精品| 亚洲精品欧美精品| 亚洲国产一区二区三区青草影视| 久久国产欧美精品| 久久久久久久综合| 好吊色欧美一区二区三区视频| 欧美在线在线| 欧美aⅴ一区二区三区视频| 亚洲黄色大片| 欧美久久久久久久| 亚洲校园激情| 亚洲精品中文在线| 欧美精品在线一区二区三区| av72成人在线| 久久久人成影片一区二区三区| 国外成人在线视频网站| 久久一区二区三区国产精品| 精品福利av| 亚洲欧美日本日韩| 久久男人资源视频| 91久久国产综合久久| 欧美激情一区在线观看| 亚洲一区二区高清| 噜噜噜噜噜久久久久久91 | 久久免费视频这里只有精品| 亚洲国产女人aaa毛片在线| 欧美日韩国产不卡| 亚洲欧美自拍偷拍| 亚洲福利视频在线| 亚洲欧美卡通另类91av| 亚洲高清影视| 国产精品毛片a∨一区二区三区| 久久成人亚洲| 99re6热只有精品免费观看| 久久精品国产综合| 99精品福利视频| 国模吧视频一区| 欧美日韩中文字幕精品| 久久久久久穴| 亚洲影院免费观看| 亚洲国产精品久久久久婷婷老年| 欧美一区二视频| 亚洲精品乱码久久久久久| 国产麻豆午夜三级精品| 欧美精品v日韩精品v国产精品 | 久久久福利视频| 亚洲精品视频在线看| 久久综合色综合88| 亚洲欧美视频一区二区三区| 亚洲黄网站黄| 国外成人在线视频网站| 国产精品素人视频| 欧美视频中文字幕在线| 欧美大片第1页| 亚洲久久一区| 欧美大学生性色视频| 久久成人免费视频| 怡红院精品视频在线观看极品| 久久香蕉国产线看观看av|