zoukankan      html  css  js  c++  java
  • Jzoj3756 【NOI2014】动物园

    下课前,园长提出了一个问题:“KMP 算法只能求出 next 数组。我现在希望求出一个更强大 num 数组——对于字符串 S 的前 i 个字符构成的子串,既是它的后缀同时又是它的前缀,并且该后缀与该前缀不重叠,将这种字符串的数量记作num[i]。例如 S 为 aaaaa,则 num[4] = 2。这是因为 S 的前 4 个字符为 aaaa,其中a 和 aa 都满足性质‘既是后缀又是前缀’,同时保证这个后缀与这个前缀不重叠。
    而 aaa 虽然满足性质‘既是后缀又是前缀’,但遗憾的是这个后缀与这个前缀重叠了,所以不能计算在内。同理,num[1] = 0,num[2] = num[3] = 1,num[5] = 2。 ”
    最后,园长给出了奖励条件,第一个做对的同学奖励巧克力一盒。听了这句话,睡了一节课的企鹅立刻就醒过来了!但企鹅并不会做这道题,于是向参观动物园的你寻求帮助。你能否帮助企鹅写一个程序求出 num 数组呢?
    特别地,为了避免大量的输出,你不需要输出 num[i] 分别是多少,你只需要输出∏ (num[i] + 1) 对 1,000,000,007 取模的结果即可。
    其中∏ (num[i] + 1)=1= (num[1] + 1) × (num[2] + 1) × ⋯ × (num[L] + 1)。

    到头来还是到卡常题。。。

    本来应该老老实实地用两次kmp的,偏偏写了一个SAM

    当然也不难写,不过是在线在parent树上面做倍增而已(插入一个做一次倍增求答案)

    我们对于新加入的节点np,只考虑它所有祖先中,sz[p]>0的节点(因为只有这些节点是串的前缀)

    倍增到一个最深的节点p使得mx[p]*2<=mx[np]即可

    因为只考虑sz[p]>0的祖先,所以求f[x][0]可以O(1),总体复杂度O(tNlogN)

    让后开始无限卡常,用尽所学手段但是卡到现在依然没过,放弃。。。。。。

    (讲道理,以前jz的机子都是跑的比我的电脑快的啊怎么今天比我这里还慢呢。。。1.2s怎么都卡不了啊)

    #pragma GCC opitmize("O2")
    #pragma G++ opitmize("O2")
    #include<stdio.h>
    #include<string.h>
    #include<algorithm>
    #define N 2000010
    #define M 1000000007
    using namespace std;
    char str[N]; long long ans=1;
    int s[N][26],mx[N],sz[N],f[N];
    int n,lst=1,cnt=1,a[20][N]; 
    inline int extend(int c){
    	register int p=lst,np=lst=++cnt,q,nq;
    	mx[np]=mx[p]+1; sz[np]=1;  memset(s[cnt],0,26<<2);
    	for(;p&&!s[p][c];p=f[p]) s[p][c]=np;
    	if(!p) { f[np]=1; goto end; }
    	q=s[p][c];
    	if(mx[q]==mx[p]+1) f[np]=q;
    	else{
    		nq=++cnt;
    		mx[nq]=mx[p]+1;
    		f[nq]=f[q]; f[q]=f[np]=nq;
    		memcpy(s[nq],s[q],26<<2);
    		for(;p&&s[p][c]==q;p=f[p]) s[p][c]=nq;
    		p=f[nq]; a[0][nq]=sz[p]?p:a[0][p];
    		for(int j=1;j<19;++j) a[j][nq]=a[j-1][a[j-1][nq]];
    	}
    	end:sz[np]+=sz[p=f[np]]; 
    	a[0][np]=sz[p]?p:a[0][p];
    	for(int j=1;j<19;++j) a[j][np]=a[j-1][a[j-1][np]];
    	p=np;
    	for(int j=18;~j;--j)
    		(mx[a[j][p]]<<1)>mx[np]?p=a[j][p]:0;
    	p=a[0][p];
    	p!=np?ans=ans*(sz[p]+1)%M:0;
    }
    int _18520(){
    	memset(sz,0,sizeof sz);
    	memset(s[1],0,26<<2); 
    	scanf("%s",str+1); n=strlen(str+1);
    	for(int i=1;i<=n;++i) extend(str[i]-'a');
    	printf("%lld
    ",ans);
    }
    int main(){
    	int T; scanf("%d",&T);
    	for(;T--;_18520())ans=cnt=lst=1;
    }

  • 相关阅读:
    (5.1.4)引擎管理——SSMS管理使用小技巧
    (5.1.3)引擎管理——恢复系统数据库
    Kafka 基本原理
    读懂基础机器学习算法
    数据挖掘常用工具
    Oracle 增加修改删除字段
    .net中使用ODP.net访问Oracle数据库(无客户端部署方法)
    数据挖掘系列(1)关联规则挖掘基本概念与Aprior算法
    ORACLE存储过程学习
    js版扫雷(可直接运行试玩)
  • 原文地址:https://www.cnblogs.com/Extended-Ash/p/9477175.html
Copyright © 2011-2022 走看看