玄武密码(bzoj4327)(JSOI2012)

题目描述

原题来自:JSOI 2012

在美丽的玄武湖畔,鸡鸣寺边,鸡笼山前,有一块富饶而秀美的土地,人们唤作进香河。相传一日,一缕紫气从天而至,只一瞬间便消失在了进香河中。老人们说,这是玄武神灵将天书藏匿在此。

很多年后,人们终于在进香河地区发现了带有玄武密码的文字。更加神奇的是,这份带有玄武密码的文字,与玄武湖南岸台城的结构有微妙的关联。于是,漫长的破译工作开始了。

经过分析,我们可以用东南西北四个方向来描述台城城砖的摆放,不妨用一个长度为 NNN 的序列来描述,序列中的元素分别是 ESWN,代表了东南西北四向,我们称之为母串。而神秘的玄武密码是由四象的图案描述而成的 MMM 段文字。这里的四象,分别是东之青龙,西之白虎,南之朱雀,北之玄武,对东南西北四向相对应。

现在,考古工作者遇到了一个难题。对于每一段文字,其前缀在母串上的最大匹配长度是多少呢?

输入格式

第一行有两个整数,NNN 和 MMM,分别表示母串的长度和文字段的个数;
第二行是一个长度为 NNN 的字符串,所有字符都满足是 ESW 和 N 中的一个;
之后 MMM 行,每行有一个字符串,描述了一段带有玄武密码的文字。依然满足,所有字符都满足是 ESW 和 N 中的一个。

输出格式

输出有 MMM 行,对应 MMM 段文字。
每一行输出一个数,表示这一段文字的前缀与母串的最大匹配串长度。

样例

样例输入

7 3
SNNSSNS
NNSS
NNN
WSEE

样例输出

4
2
0

数据范围与提示

对于全部数据,1≤N≤10^7,1≤M≤10^5,保证每一段文字的长度均小于等于 100。


题目描述和题其实莫得关系

上来很自然的想到了KMP

然后发现O(n*100m)会T

然后发现了AC自动机的正解,很SB的写了一上午(最主要的是教练一直说这次初赛分数线提高了,慌得一批)

首先我们建树,然后把主串放在上面跑,很显然,对于每一个前缀如果能够成功匹配,那么它的nxt肯定也可以

对于每一个可以匹配的地方,我们标记一次,最后对于每一个模式串都从后往前跑一遍,当我们发现一个地方被标记后,就输出答案(从后往前跑所找到的第一个一定是最长的能匹配的前缀)

还有就是这道题的数据范围很迷,数组一定要开够

下面给出代码:

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<string>
#include<cmath>
using namespace std;
inline int rd(){
    int x=0,f=1;
    char ch=getchar();
    for(;!isdigit(ch);ch=getchar()) if(ch==‘-‘) f=-1;
    for(;isdigit(ch);ch=getchar()) x=x*10+ch-‘0‘;
    return x*f;
}
inline void write(int x){
    if(x<0) putchar(‘-‘),x=-x;
    if(x>9) write(x/10);
    putchar(x%10+‘0‘);
    return ;
}
int n,m;
char s[10000006],a[106];
int trie[1000006][4];
int len[100006],f[10000006];
int vis[10000006];
int tot=1;
int calc(char ch){
    if(ch==‘E‘) return 0;
    if(ch==‘W‘) return 1;
    if(ch==‘N‘) return 2;
    if(ch==‘S‘) return 3;
}
void pre(int x){
    int c=1;
    for(int i=1;i<=len[x];i++){
        int h=calc(a[i]);
        if(!trie[c][h]){
            f[++tot]=c;
            trie[c][h]=tot;
        }
        c=trie[c][h];
    }
    vis[x]=c;
    return ;
}
int q[1000006];
int l=0,r=0;
int nxt[1000006];
void get_next(){
    q[++r]=1;
    for(int i=0;i<=3;i++) trie[0][i]=1;
    nxt[1]=0;
    while(l<r){
        int h=q[++l];
        for(int i=0;i<=3;i++){
            if(!trie[h][i]) trie[h][i]=trie[nxt[h]][i];
            else{
                nxt[trie[h][i]]=trie[nxt[h]][i];
                q[++r]=trie[h][i];
            }
        }
    }
    return ;
}
bool book[1000006];
void solve(){
    int c=1;
    for(int i=1;i<=n;i++){
        int h=calc(s[i]);
        c=trie[c][h];
        for(int j=c;j;j=nxt[j]){
            if(book[j]) break;
            book[j]=1;
        }
    }
    return ;
}
int gets(int x){
    int ans=len[x];
    for(int i=vis[x];i;i=f[i],ans--) if(book[i]) return ans;
}
int main(){
    n=rd(),m=rd();
    scanf("%s",s+1);
    for(int i=1;i<=m;i++){
        scanf("%s",a+1);
        len[i]=strlen(a+1);
        pre(i);
    }
    get_next();
    solve();
    for(int i=1;i<=m;i++) printf("%d\n",gets(i));
    return 0;
}

原文地址:https://www.cnblogs.com/WWHHTT/p/9749628.html

时间: 2024-10-27 19:47:14

玄武密码(bzoj4327)(JSOI2012)的相关文章

【bzoj4327】JSOI2012 玄武密码 AC自动机

题目描述 在美丽的玄武湖畔,鸡鸣寺边,鸡笼山前,有一块富饶而秀美的土地,人们唤作进香河.相传一日,一缕紫气从天而至,只一瞬间便消失在了进香河中.老人们说,这是玄武神灵将天书藏匿在此. 很多年后,人们终于在进香河地区发现了带有玄武密码的文字.更加神奇的是,这份带有玄武密码的文字,与玄武湖南岸台城的结构有微妙的关联.于是,漫长的破译工作开始了. 经过分析,我们可以用东南西北四个方向来描述台城城砖的摆放,不妨用一个长度为N的序列来描述,序列中的元素分别是‘E’,‘S’,‘W’,‘N’,代表了东南西北四

【AC自动机】bzoj4327: JSOI2012 玄武密码

题目思路没话讲:主要是做题时候的细节和经验 Description 在美丽的玄武湖畔,鸡鸣寺边,鸡笼山前,有一块富饶而秀美的土地,人们唤作进香河.相传一日,一缕紫气从天而至,只一瞬间便消失在了进香河中.老人们说,这是玄武神灵将天书藏匿在此. 很多年后,人们终于在进香河地区发现了带有玄武密码的文字.更加神奇的是,这份带有玄武密码的文字,与玄武湖南岸台城的结构有微妙的关联.于是,漫长的破译工作开始了. 经过分析,我们可以用东南西北四个方向来描述台城城砖的摆放,不妨用一个长度为N的序列来描述,序列中的

AC自动机 --- [JSOI2012]玄武密码

bzoj 4327 [JSOI2012]玄武密码 题目描述 在美丽的玄武湖畔,鸡鸣寺边,鸡笼山前,有一块富饶而秀美的土地,人们唤作进香河. 相传一日,一缕紫气从天而至,只一瞬间便消失在了进香河中.老人们说,这是玄武神灵将天书藏匿在此. 很多年后,人们终于在进香河地区发现了带有玄武密码的文字. 更加神奇的是,这份带有玄武密码的文字,与玄武湖南岸台城的结构有微妙的关联.于是,漫长的破译工作开始了. 经过分析,我们可以用东南西北四个方向来描述台城城砖的摆放,不妨用一个长度为N的序列来描述,序列中的元素

「JSOI2012」玄武密码

「JSOI2012」玄武密码 传送门 题目是要求多个串在母串上的最长匹配长度. 考虑 \(\text{AC}\) 自动机,我们建出 \(\text{Trie}\) 图然后用母串来在上面跑. 每一个能匹配的位置,它 \(\text{fail}\) 的位置也一定可以匹配,我们就跳 \(\text{fail}\) 把经过的节点的值赋为 \(1\) ,表示这个位置可以匹配. 然后我们每一个模式串,找到ta在 \(\text{Trie}\) 树上最深的可以匹配的节点来计算答案即可. 参考代码: #incl

【AC自动机】玄武密码

[题目链接] https://loj.ac/problem/10058 [题意] 对于每一段文字,其前缀在母串上的最大匹配长度是多少呢 [参考别人的题解] https://www.luogu.org/problemnew/solution/P5231 我们只需要先建立所有密码的trie树再以母串为主串跑一个AC自动机不过其中还是有一些需要改动的地方原本字典树中用来记录某个节点是不是字符串结尾的数组不需要,直接删去我们需要另一个数组来标记哪些点被匹配跑完ac自动机后从trie树上找最后一个匹配的点

TZOJ 5986 玄武密码(AC自动机)

描述 在美丽的玄武湖畔,鸡鸣寺边,鸡笼山前,有一块富饶而秀美的土地,人们唤作进香河.相传一日,一缕紫气从天而至,只一瞬间便消失在了进香河中.老人们说,这是玄武神灵将天书藏匿在此. 很多年后,人们终于在进香河地区发现了带有玄武密码的文字.更加神奇的是,这份带有玄武密码的文字,与玄武湖南岸台城的结构有微妙的关联.于是,漫长的破译工作开始了. 经过分析,我们可以用东南西北四个方向来描述台城城砖的摆放,不妨用一个长度为N的序列来描述,序列中的元素分别是‘E’,‘S’,‘W’,‘N’,代表了东南西北四向,

BZOJ4327 : JSOI2012 玄武密码

对所有询问串建立AC自动机. 然后将母串在AC自动机上跑,每走到一个点x,从x点出发沿着fail指针能到的所有前缀都是匹配成功的,暴力向上走,碰到走过的就break,这样每个点最多只会被标记一次. 时间复杂度$O(N+100M)$. #include<cstdio> #include<cstring> const int N=10000003,M=100010; int tot,son[N][4],f[N],fail[N],q[N];bool v[N]; int n,m,i,fin

vijos-1951 玄武密码

题意: 给出一个匹配串和n个单词: 求每个单词在匹配串中出现的的最大前缀长度: 匹配串长度<=10^7,n<=10^5,单词长度<=100: 题解: 当年啥也不会天真的一发KMP骗掉了50分,然后看题解说是自动机感觉好神啊: 现在回来复习自动机就把这道题切了试试: 基本的建立自动机什么的不说了: 主要就是答案的处理上我是在trie树上记录一个is的数组: 然后每个和匹配串匹配到了的结点全都标记上: (当然这里要把fail指针的一连串的后缀相同的结点标记) 最后和建树时一样扫一遍自动机,每

[JSOI 2012] 玄武密码

[题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=4327 [算法] AC自动机[代码] #include<bits/stdc++.h> using namespace std; const int MAXN = 1e7 + 10; const int MAXM = 1e5 + 10; const int MAXLEN = 110; int n,m; int ans[MAXM]; char P[MAXN]; char s[MAXM]