Description

(我并不想告诉你题目名字是什么鬼)

有一个长度为n的仅包含小写字母的字符串S,下标范围为[1,n].

现在有若干组询问,对于每一个询问,我们给出若干个后缀(以其在S中出现的起始位置来表示),求这些后缀两两之间的LCP(LongestCommonPrefix)的长度之和.一对后缀之间的LCP长度仅统计一遍.

Input

第一行两个正整数n,m,分别表示S的长度以及询问的次数.

接下来一行有一个字符串S.

接下来有m组询问,对于每一组询问,均按照以下格式在一行内给出:

首先是一个整数t,表示共有多少个后缀.接下来t个整数分别表示t个后缀在字符串S中的出现位置.

Output

对于每一组询问,输出一行一个整数,表示该组询问的答案.由于答案可能很大,仅需要输出这个答案对于23333333333333333(一个巨大的质数)取模的余数.

题解:和CF的一个题几乎一模一样,都是利用虚树统计答案

#include <bits/stdc++.h>
#define setIO(s) freopen(s".in","r",stdin) , freopen(s".out","w",stdout)
#define maxn 1200004
#define mod 23333333333333333
#define ll long long
using namespace std;
int edges,n,Q;
int hd[maxn],to[maxn],nex[maxn],tr[maxn];
char str[maxn];
void addedge(int u,int v,int c) //1001 个
{
nex[++edges]=hd[u],hd[u]=edges,to[edges]=v;
}
namespace SAM
{
int last, tot;
int len[maxn<<1],f[maxn<<1],trans[maxn<<1][27];
void init() { last=tot=1; }
int extend(int c)
{
int np=++tot,p=last;
len[np]=len[p]+1,last=np;
while(p&&!trans[p][c]) trans[p][c]=np, p=f[p];
if(!p) f[np]=1;
else
{
int q=trans[p][c];
if(len[q]==len[p]+1) f[np]=q;
else
{
int nq=++tot;
len[nq]=len[p]+1;
memcpy(trans[nq],trans[q],sizeof(trans[q]));
f[nq]=f[q],f[q]=f[np]=nq;
while(p&&trans[p][c]==q) trans[p][c]=nq,p=f[p];
}
}
return np;
}
void build()
{
for(int i=2;i<=tot;++i) addedge(f[i],i,len[f[i]]-len[i]);
}
}
int tim;
int dfn[maxn],Top[maxn],hson[maxn],siz[maxn],fa[maxn],dis[maxn];
void dfs1(int u,int ff)
{
fa[u]=ff,siz[u]=1,dfn[u]=++tim,dis[u]=dis[ff]+1;
for(int i=hd[u];i;i=nex[i])
{
int v=to[i];
if(v==ff) continue;
dfs1(v,u);
siz[u]+=siz[v];
if(siz[v]>siz[hson[u]]) hson[u]=v;
}
}
void dfs2(int u,int tp)
{
Top[u]=tp;
if(hson[u]) dfs2(hson[u],tp);
for(int i=hd[u];i;i=nex[i])
{
int v=to[i];
if(v==fa[u]||v==hson[u]) continue;
dfs2(v,v);
}
}
int LCA(int x,int y)
{
while(Top[x]!=Top[y])
{
dis[Top[x]]>dis[Top[y]]?x=fa[Top[x]]:y=fa[Top[y]];
}
return dis[x]<dis[y]?x:y;
}
vector<int>G[maxn<<1];
ll ans=0;
int t=0;
int A[maxn<<1],mk[maxn<<1],size[maxn<<1],S[maxn<<1];
bool cmp(int i,int j)
{
return dfn[i]<dfn[j];
}
void addvir(int x,int y)
{
G[x].push_back(y);
}
void insert(int x)
{
if(t<=1) { S[++t]=x; return; }
int lca=LCA(S[t],x);
if(lca==S[t]) { S[++t]=x; return; }
while(t>1&&dis[S[t-1]]>=dis[lca]) addvir(S[t-1],S[t]),--t;
if(S[t]!=lca) addvir(lca,S[t]),S[t]=lca;
S[++t]=x;
}
void DP(int x)
{
size[x]=mk[x];
for(int i=0;i<G[x].size();++i)
{
DP(G[x][i]);
ans+=1ll*size[x]*size[G[x][i]]%mod*SAM::len[x]%mod;
ans%=mod;
size[x]+=size[G[x][i]];
}
G[x].clear();
mk[x]=0;
}
void work()
{
int k,a;
scanf("%d",&k);
for(int i=1;i<=k;++i) scanf("%d",&a), A[i]=tr[a], mk[A[i]]=1;
sort(A+1,A+1+k,cmp);
k=unique(A+1,A+1+k)-(A+1);
t=ans=0;
if(A[1]!=1) S[++t]=1;
for(int i=1;i<=k;++i) insert(A[i]);
while(t>1) addvir(S[t-1],S[t]),--t;
DP(1);
printf("%lld\n",ans);
}
int main()
{
// setIO("input");
scanf("%d%d%s",&n,&Q,str+1);
SAM::init();
for(int i=n;i>=1;--i) tr[i]=SAM::extend(str[i]-'a');
SAM::build();
dis[1]=1,dfs1(1,0),dfs2(1,1);
for(int i=1;i<=Q;++i)
{
work();
}
return 0;
}

  

BZOJ 3879: SvT 虚树 + 后缀自动机的更多相关文章

  1. BZOJ 3879: SvT [虚树 后缀树]

    传送门 题意: 多次询问,给出一些后缀,求两两之间$LCP$之和 哈哈哈哈哈哈哈竟然$1A$了,刚才还在想如果写不好这道题下节数学就不上了,看来是上天让我上数学课啊 $Suffix\ Virtual\ ...

  2. 回文树&后缀自动机&后缀数组

    KMP,扩展KMP和Manacher就不写了,感觉没多大意思.   之前感觉后缀自动机简直可以解决一切,所以不怎么写后缀数组.   马拉车主要是通过对称中心解决问题,有的时候要通过回文串的边界解决问题 ...

  3. [HEOI2016/TJOI2016]字符串(后缀数组+二分+主席树/后缀自动机+倍增+线段树合并)

    后缀数组解法: 先二分最长前缀长度 \(len\),然后从 \(rnk[c]\) 向左右二分 \(l\) 和 \(r\) 使 \([l,r]\) 的 \(height\geq len\),然后在主席树 ...

  4. 【BZOJ-4556】字符串 后缀数组+二分+主席树 / 后缀自动机+线段树合并+二分

    4556: [Tjoi2016&Heoi2016]字符串 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 657  Solved: 274[Su ...

  5. HackerRank Special Substrings 回文树+后缀自动机+set

    传送门 既然要求对每个前缀都求出答案,不难想到应该用回文树求出所有本质不同的回文子串. 然后考虑如何对这些回文子串的前缀进行去重. 结论:答案等于所有本质不同的回文子串长之和减去字典序相邻的回文子串的 ...

  6. BZOJ 3238 [Ahoi2013]差异(后缀自动机)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3238 [题目大意] 给出一个串,设T[i]表示从第i位开始的后缀, 求sum(len( ...

  7. bzoj 3879: SvT

    Description (我并不想告诉你题目名字是什么鬼) 有一个长度为n的仅包含小写字母的字符串S,下标范围为[1,n]. 现在有若干组询问,对于每一个询问,我们给出若干个后缀(以其在S中出现的起始 ...

  8. HDU - 6704 K-th occurrence (后缀数组+主席树/后缀自动机+线段树合并+倍增)

    题意:给你一个长度为n的字符串和m组询问,每组询问给出l,r,k,求s[l,r]的第k次出现的左端点. 解法一: 求出后缀数组,按照排名建主席树,对于每组询问二分或倍增找出主席树上所对应的的左右端点, ...

  9. BZOJ 2286 消耗战 (虚树+树形DP)

    给出一个n节点的无向树,每条边都有一个边权,给出m个询问,每个询问询问ki个点,问切掉一些边后使得这些顶点无法与顶点1连接.最少的边权和是多少.(n<=250000,sigma(ki)<= ...

随机推荐

  1. VMware 虚拟化编程(3) —VMware vSphere Web Service API 解析

    目录 目录 前文列表 VMware vSphere Web Services API VMware vSphere Web Services SDK vSphere WS API 中的托管对象 Man ...

  2. 删除历史日志的一个API

    删除历史日志的一个API bool DeleteOldFiles(const char* strFolder, const char* strPrefix, bool is_recursion, UI ...

  3. oracle--多表联合查询sql92版

    sql92学习 -查询员工姓名,工作,薪资,部门名称 sql的联合查询(多表查询) --1.sql92标准 ----笛卡尔积:一件事情的完成需要很多步骤,而不同的步骤有很多种方式,完成这件事情的所有方 ...

  4. Hibernate入门4

    HIbernate的导航查询: 适用场景:当一张A表关联到另一张B表的多条记录,存在一对多的关系(或者多对多),那么查询A表的记录时,就可以将A表某条记录关联的B表的所有记录查询出来,这种方式,就叫做 ...

  5. PyTorch笔记之 squeeze() 和 unsqueeze()

    1.squeeze() 函数 squeeze() 用来去掉向量的一个维度,只有维度为 1 的那一维才能去掉 example: 初始化1个向量shape为(1,2,3)的向量 import torch ...

  6. 解决浏览器打开网页后提示“dns_probe_possible”的方法

    使用浏览器浏览网页时偶尔会遇到无法上网且浏览器提示:DNS_PROBE_POSSIBLE 一般有三种情况会导致这样的故障: 1.网络协议出现故障,也就是常说的 DNS 设置问题 2.浏览器中设置问题, ...

  7. 解决在Tomcat中的server.xml中修改了配置,启动后还原的问题

    正确答案,你要在eclipse的项目Servers中,找到你的tomcat,打开有各种配置文件,在这里改,才会在使用eclipse启动tomcat后不会自动恢复.

  8. 使用pdfobject.js实现在线浏览PDF

    1.pdfobject.js官网:https://pdfobject.com/ 2.在html文件中引入这个文件,以pdfobject.min.js为例 <script type="t ...

  9. Linux安装Python3以及虚拟环境

    python3的linux环境编译安装 1.linux下安装软件的方式 选则yum工具,方便,自行解决软件之间的依赖关系,自动下载且安装 1.配置yum源 可以选择阿里云源,清华源等 配置第一个仓库, ...

  10. 小白学Python(14)——pyecharts 绘制K线图 Kline/Candlestick

    Kline-基本示例 from pyecharts import options as opts from pyecharts.charts import Kline data = [ [2320.2 ...