这题就比较有趣了。

首先匹配一遍,然后统计子树叶子节点中包含大于等于k的节点个数(HH的项链)

然后就可以搞了。

关于合法的情况数,显然是l[i]-l[fa[i]],然后向下下传即可(YY一下)。

#include <vector>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std; #define F(i,j,k) for (int i=j;i<=k;++i)
#define D(i,j,k) for (int i=j;i>=k;--i)
#define ll long long
#define maxn 600005
struct Query{
int l,r,id;
void print(){printf("Query : %d to %d (id is %d)\n",l,r,id);}
}brr[maxn];
bool cmp(Query a,Query b)
{return a.l==b.l?a.r<b.r:a.l<b.l;}
struct Bit_Tree{
int x[maxn];
void add(int pos,int f,int tot)
{for (;pos<=tot;pos+=(pos&(-pos))) x[pos]+=f;}
int query(int pos)
{int ret=0;for(;pos;pos-=(pos&(-pos)))ret+=x[pos];return ret;}
}BT;
class Suffix_Automata{
public:
int arr[maxn],tot,top;
int last,cnt,n,k;
char s[maxn];
vector <int> v[maxn],vb[maxn];
int h[maxn],to[maxn],ne[maxn],en,in[maxn],out[maxn];
int go[maxn][26],l[maxn],fa[maxn];
void init()
{
last=cnt=1; en=0;
memset(h,-1,sizeof h);
memset(go,0,sizeof go);
}
void add(int x,int id)
{
int p=last,q;
if (q=go[p][x])
{
if (l[q]==l[p]+1) last=q;
else
{
int nq=++cnt;
l[nq]=l[p]+1;
memcpy(go[nq],go[q],sizeof go[q]);
fa[nq]=fa[q];
fa[q]=nq;
for(;p&&go[p][x]==q;p=fa[p]) go[p][x]=nq;
last=nq;
}
}
else
{
int np=++cnt; l[np]=l[p]+1;
for (;p&&!go[p][x];p=fa[p]) go[p][x]=np;
if (!p) fa[np]=1;
else
{
q=go[p][x];
if (l[q]==l[p]+1) fa[np]=q;
else
{
int nq=++cnt;
l[nq]=l[p]+1;
memcpy(go[nq],go[q],sizeof go[q]);
fa[nq]=fa[q];
fa[q]=fa[np]=nq;
for (;p&&go[p][x]==q;p=fa[p]) go[p][x]=nq;
}
}
last=np;
}
v[last].push_back(id);
vb[id].push_back(last);
}
void addedge(int a,int b)
{
to[en]=b;
ne[en]=h[a];
h[a]=en++;
}
void ins(int id)
{
last=1;scanf("%s",s+1);
int len=strlen(s+1);
F(j,1,len) add(s[j]-'a',id);
}
void dfs(int o)
{
in[o]=++tot;
for (int i=0;i<v[o].size();++i) arr[tot++]=v[o][i];
for (int i=h[o];i>=0;i=ne[i]) dfs(to[i]);
out[o]=tot;
++top;brr[top].l=in[o];brr[top].r=out[o];brr[top].id=o;
}
void dfs2(int o)
{
for (int i=h[o];i>=0;i=ne[i])
f[to[i]]+=f[o],dfs2(to[i]);
}
int lst[maxn],nxt[maxn],ans[maxn],f[maxn];
void solve()
{
init();
scanf("%d%d",&n,&k);
F(i,1,n) ins(i);
F(i,1,cnt) addedge(fa[i],i);
dfs(1);
// printf("Arr : ");F(i,1,tot) printf("%d ",arr[i]); printf("\n");
sort(brr+1,brr+top+1,cmp);
// F(i,1,top) brr[i].print();
F(i,1,tot)
{
if (!lst[arr[i]]&&arr[i]) BT.add(i,1,tot);
else nxt[lst[arr[i]]]=i;
lst[arr[i]]=i;
}
// F(i,1,tot) printf("%d ",nxt[i]); printf("\n");
int now=1;
F(i,1,top)
{
while (now<brr[i].l)
{
if (nxt[now]&&arr[nxt[now]]) BT.add(nxt[now],1,tot);
now++;
}
ans[brr[i].id]=BT.query(brr[i].r)-BT.query(brr[i].l-1);
}
// F(i,1,top) printf("[%d] %d\n",i,ans[i]);
F(i,1,top) if (ans[i]>=k) f[i]=l[i]-l[fa[i]]; else f[i]=0;
dfs2(1);
F(i,1,n)
{
ll ans=0;
F(j,0,vb[i].size()-1) ans+=f[vb[i][j]];
printf("%lld",ans);
if (i!=n) printf(" ");
}
}
}sam; int main()
{sam.solve();}

  

BZOJ 3473 字符串 ——广义后缀自动机的更多相关文章

  1. BZOJ 3473: 字符串 [广义后缀自动机]

    3473: 字符串 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 354  Solved: 160[Submit][Status][Discuss] ...

  2. BZOJ 3277 串 & BZOJ 3473 字符串 (广义后缀自动机、时间复杂度分析、启发式合并、线段树合并、主席树)

    标签那么长是因为做法太多了... 题目链接: (bzoj 3277) https://www.lydsy.com/JudgeOnline/problem.php?id=3277 (bzoj 3473) ...

  3. bzoj 3277: 串 & bzoj 3473: 字符串【后缀自动机||后缀数组】

    建一个广义后缀自动机(每加完一个串都返回root),在parent树上dpsum记录合法长度,打着时间戳往上跳,最后每个串在自动机上跑一变统计答案即可. 后缀数组理解起来可能方便一点,但是难写,就只说 ...

  4. BZOJ 3277 串 (广义后缀自动机)

    3277: 串 Time Limit: 10 Sec Memory Limit: 128 MB Submit: 309 Solved: 118 [Submit][Status][Discuss] De ...

  5. 【bzoj3277/bzoj3473】串/字符串 广义后缀自动机

    题目描述 字符串是oi界常考的问题.现在给定你n个字符串,询问每个字符串有多少子串(不包括空串)是所有n个字符串中至少k个字符串的子串(注意包括本身). 输入 第一行两个整数n,k.接下来n行每行一个 ...

  6. BZOJ 2894: 世界线 广义后缀自动机

    Code: #include<bits/stdc++.h> #define maxn 300000 #define ll long long using namespace std; ve ...

  7. bzoj 3926 转换+广义后缀自动机

    思路:重点在于叶子节点只有20个,我们把叶子节点提到根,把20个trie图插入后缀自动机,然后就是算有多少个本质不同的字串. #include<bits/stdc++.h> #define ...

  8. BZOJ 4180: 字符串计数 后缀自动机 + 矩阵乘法 + 二分(神题)

    Description SD有一名神犇叫做Oxer,他觉得字符串的题目都太水了,于是便出了一道题来虐蒟蒻yts1999.   他给出了一个字符串T,字符串T中有且仅有4种字符 'A', 'B', 'C ...

  9. BZOJ3473 字符串 广义后缀自动机

    今天主攻了下SAM 好多东西以前都没理解到 对于这道题 我们建一个自动机存所有串 每个穿last从1开始 对于自动机上每个点额外记一个cnt 表示能匹配到这个点的不同串个数 建完对每个串在自动机上匹配 ...

随机推荐

  1. httpmodule初识

    .net的请求流程: HttpRequest-->inetinfo.exe->ASPNET_ISAPI.DLL-->Http Pipeline-->ASPNET_WP.EXE- ...

  2. vue2.0的变化

    1. 在每个组件模板,不在支持片段代码 组件中模板: 之前: <template> <h3>我是组件</h3><strong>我是加粗标签</st ...

  3. C#中Json进行序列化时去掉值为null的节点

    当我们用json文件为数据源时,并对json数据进行操作时可能会产生一些数值为null的节点生成,想要去掉null的节点需要一些操作 本文用一个简单的工具对json进行操作 工具:Newtonsoft ...

  4. const、let、var的区别

    const不能从字面上来理解,他不能修改的是栈内存在的值和地址. 使用const声明的是常量,在后面出现的代码中不能再修改该常量的值. 怎么理解栈内存在的值和地址呢?就要从javascript的类型说 ...

  5. C++函数调用过程深入分析<转>

    转自http://blog.csdn.net/dongtingzhizi/article/details/6680050 C++函数调用过程深入分析 作者:靠谱哥 微博:洞庭之子-Bing 0. 引言 ...

  6. CPP-基础:运算符重载详解

    1.运算符重载定义: C++中预定义的运算符的操作对象只能是基本数据类型.但实际上,对于许多用户自定义类型(例如类),也需要类似的运算操作.这时就必须在C++中重新定义这些运算符,赋予已有运算符新的功 ...

  7. vs 2017 boost 安装目录 非安装

    linuxg++ -Wall -std=c++11 boost_socks5.cpp -o boost_socks5 -lboost_system -lboost_thread -lpthread m ...

  8. 【转载】K-mer算法

    k-mer是指将reads分成包含k个碱基的字符串,一般长短为m的reads可以分成m-k+1个k-mers.举个例子吧,为了简化,有这么个reads(当然实际比这个长):AACTGACTGA.如果k ...

  9. java中regex参考

    在Sun的Java JDK 1.40版本中,Java自带了支持正则表达式的包,本文就抛砖引玉地介绍了如何使用java.util.regex包. 可粗略估计一下,除了偶尔用Linux的外,其他Linu ...

  10. Java中的线程--线程中的工具

    这主要想写一下Java中的jdk提供的一些线程中的工具, 一.semaphore信号灯 Semaphore可以维护当前访问自身的线程个数,并提供了同步机制,使用Semaphore可以控制同时访问资源的 ...