CENSORING
题目描述
FJ为它的奶牛订阅了很多杂志,balabala.......,其中有一些奶牛不宜的东西(比如如何煮牛排)。
FJ将杂志中所有的文章提取出来组成一个长度最多为10^5的字符串S。他有一个要从S中删除的词语的列表,t1,t2...tn。
FJ每次找到最早的出现在列表里的子串,然后将其删去。他重复此过程,直到找不到这样的子串。值得注意的是删除一个单词可能产生一个新的之前并没有出现过的要被删除的单词。
FJ保证列表中没有一个字符串是另一个字符串的子串。要就是说每次要删的单词是唯一确定的。
帮助FJ确定最终S的删减版。
Solution
考虑把T建成AC自动机。把S扔进去匹配,,用栈记录每一步的位置,遇到可匹配点就往后跳到匹配该串前的位置。
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
#define maxn 300050
using namespace std;
int n,len,tot,top;
int tr[maxn][],fail[maxn],p[maxn];
char s[maxn],ch[maxn];
int zh[maxn];char ans[maxn];
void ins(){
int len=strlen(ch),k=;
for(int i=;i<len;i++){
if(!tr[k][ch[i]-'a'])tr[k][ch[i]-'a']=++tot;
k=tr[k][ch[i]-'a'];
}
p[k]=len;
}
void build(){
queue<int>q;
for(int i=;i<;i++)if(tr[][i])q.push(tr[][i]);
while(!q.empty()){
int k=q.front();q.pop();
for(int i=;i<;i++){
if(tr[k][i]){
fail[tr[k][i]]=tr[fail[k]][i];
q.push(tr[k][i]);
}
else tr[k][i]=tr[fail[k]][i];
}
}
}
int main()
{
scanf("%s",s);
cin>>n;
for(int i=;i<=n;i++){
scanf("%s",ch);
ins();
}
build();
n=strlen(s);int k=;
for(int i=;i<n;i++){
k=tr[k][s[i]-'a'];
zh[++top]=k;ans[top]=s[i];
if(p[k]){
top-=p[k];k=zh[top];
}
}
for(int i=;i<=top;i++)printf("%c",ans[i]);
return ;
}