BZOJ 12 HNOI 2004 L 언어 AC 자동기(Trie 트리) + 동적 기획
더 이상 데이터 범위를 잘못 보지 못하겠어...트리로 해결할 수 있는 문제를 AC자동기로 썼어...
AC 로봇의 각 단어가 있는 노드에 단어표의 단어를 모두 삽입하여 해당 단어의 길이를 기록합니다.
그리고 모든 문자열에 대해 f[i]로 길이가 i인 접두사를 단어표의 단어 달리기 AC 자동기로 분할할 수 있는지 여부를 표시합니다
일치하는 노드마다 이 노드에서 루트로의fail 경로에 있는 모든len f[i]|=f[i-len]
가장 큰 f[i]를 찾으면 답입니다.
단어 길이가 최대 10이니까 트리로 바로 폭력을 가하면 돼요.
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define M 1050000
using namespace std;
struct Trie{
int len;
Trie *fail,*son[26];
void* operator new (size_t size);
}*root,*mempool,*C;
int n,m;
char s[M];
void* Trie :: operator new (size_t size)
{
if(C==mempool)
{
C=new Trie[1<<15];
mempool=C+(1<<15);
memset(C,0,sizeof(Trie)*(1<<15) );
}
return C++;
}
void Insert(Trie*&p,char *pos,int dpt)
{
if(!p) p=new Trie;
if(!*pos)
{
p->len=dpt;
return ;
}
Insert(p->son[(*pos)-'a'],pos+1,dpt+1);
}
void BFS()
{
static Trie* q[1<<16];
static unsigned short r,h;
int i;Trie *temp;
for(i=0;i<26;i++)
if(temp=root->son[i])
{
temp->fail=root;
q[++r]=temp;
}
while(r!=h)
{
Trie *p=q[++h];
for(i=0;i<26;i++)
if(p->son[i])
{
temp=p->fail;
while( temp!=root && !temp->son[i] )
temp=temp->fail;
if( temp->son[i] )
temp=temp->son[i];
p->son[i]->fail=temp;
q[++r]=p->son[i];
}
}
}
int Aho_Corasick_Automaton()
{
int i,re=0;
Trie *p=root,*temp;
static bool f[M];f[0]=1;
for(i=1;s[i];i++)
{
f[i]=0;
while( p!=root && !p->son[s[i]-'a'] )
p=p->fail;
if(p->son[s[i]-'a'])
{
p=p->son[s[i]-'a'];
for(temp=p;temp!=root;temp=temp->fail)
if(temp->len)
{
f[i]|=f[i-temp->len];
if(f[i]) break;
}
}
if(f[i]) re=i;
}
return re;
}
int main()
{
int i;
cin>>n>>m;
for(i=1;i<=n;i++)
scanf("%s",s+1),Insert(root,s+1,0);
BFS();
for(i=1;i<=m;i++)
{
scanf("%s",s+1);
cout<<Aho_Corasick_Automaton()<<endl;
}
}
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
01 가방, 완전 가방, 다중 가방 dp(동적 기획 입문 dp)01 가방은 2진법으로 직접 표시할 수 있지만 데이터 양이 너무 많으면 시간을 초과하는 것이 폭력이다.01 가방의 사상은 바로 이 물품에 대해 내가 넣은 가치가 큰지 안 넣은 가치가 큰지 비교하여 방정식 f[i][v...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.