欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

[luogu2292][L语言]

程序员文章站 2022-05-02 10:07:36
思路 这道题我用的是AC自动机的做法。 先把子串挂到tried树上,在单词结尾打标记的时候,标记的是当前单词的长度。然后去上面查询母串的时候,每查询到一个单词,就建立一条 ......

题目链接

思路

这道题我用的是ac自动机的做法。

先把子串挂到trie树上,在单词结尾打标记的时候,标记的是当前单词的长度。然后去上面查询母串的时候,每查询到一个单词,就建立一条线段,这条线段的结尾位置是母串当前的位置,开始位置就是用当前位置减去这个单词的长度。

然后只要去判断,选出一些线段,使得这些线段紧挨着(既不相互覆盖,中间也不遗漏)从母串开始位置铺,能铺到的最远的地方。所以只需要一遍bfs就行了。

代码

/*
* @author: wxyww
* @date:   2018-12-17 08:38:38
* @last modified time: 2018-12-17 09:09:28
*/
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cmath>
#include<ctime>
#include<bitset>
#include<cstring>
#include<queue>
using namespace std;
typedef long long ll;
const int n = 2000000 + 100;
ll read() {
   ll x=0,f=1;char c=getchar();
   while(c<'0'||c>'9') {
      if(c=='-') f=-1;
      c=getchar();
   }
   while(c>='0'&&c<='9') {
      x=x*10+c-'0';
      c=getchar();
   }
   return x*f;
}
int trie[1000][30],end[1000];
char s[30];
int tot;
void insert() {
   int len = strlen(s + 1);
   int now = 0;
   for(int i = 1;i <= len;++i) {
      int k = s[i] - 'a';
      if(!trie[now][k]) trie[now][k] = ++tot;
      now = trie[now][k];
   }
   end[now] = len;
}
struct node {
   int l,r;
}e[n];
int ejs;
int fail[n];
char s[n];
queue<int>q;
void build() {
   for(int i = 0;i < 26;++i) if(trie[0][i]) q.push(trie[0][i]);
   while(!q.empty()) {
      int u = q.front();q.pop();
      for(int i = 0;i < 26;++i) {
         if(trie[u][i]) fail[trie[u][i]] = trie[fail[u]][i],q.push(trie[u][i]);
         else trie[u][i] = trie[fail[u]][i];
      }
   }
}
vector<int>v[n];
void work() {
   int len = strlen(s + 1);
   ejs = 0;int now = 0;
   for(int i = 1;i <= len;++i) {
      now = trie[now][s[i] - 'a'];
      for(int j = now;j;j = fail[j]) {
         if(end[j]) e[++ejs].l = i - end[j] + 1,e[ejs].r = i;
      }
   }
}
int vis[n];
void bfs() {
   q.push(1);
   int ans = 0;
   for(int i = 1;i <= ejs;++i) v[e[i].l].push_back(i);
   memset(vis,0,sizeof(vis));
   while(!q.empty()) {
      int l = q.front();q.pop();
      int k = v[l].size();
      for(int i = 0;i < k;++i) {
         int z = v[l][i];
         if(vis[z]) continue;
         vis[z] = 1;
         q.push(e[z].r + 1);
         ans = max(ans,e[z].r);
      }
   }
   printf("%d\n",ans);
   int len = strlen(s + 1);
   for(int i = 1;i <= len;++i) v[i].clear();
}
int main() {
   int n = read(),m = read();
   for(int i = 1;i <= n;++i) {
      scanf("%s",s + 1);
      insert();
   }
   build();
   for(int i = 1;i <= m;++i) {
      scanf("%s",s + 1);
      work();
   bfs();
   }

   return 0;
}