正题

题目链接:https://www.luogu.com.cn/problem/P7046


题目大意

给出一个长度为 \(n\) 的字符串,然后 \(m\) 次把它的一个子串加入集合。如果一个字符串在这个集合中作为字符串的后缀出现次数大于 \(k\) 那么这个字符串就会被计入贡献。

每次求计入贡献的字符串数和最长长度。

\(1\leq n,m\leq 5\times 10^5,0\leq k<n\)。


解题思路

考虑在parents树上如果能定位到一个节点的字符串那么它的后缀就是它到根的路径。

但是可能定位不到根,一种暴力的做法是每条边上建一个线段树然后暴力改。但是这样很麻烦可以考虑让每个询问一定能定位到一个节点。

我们直接建好树然后每次把询问倍增挂到对应的边上用set储存,然后再重新建一棵包含每个节点的树。

那么现在问题就变为了统计子树权值大于 \(k\) 的节点了,因为每个点到根的路径上满足条件的边一定是一段后缀,而每个节点最多统计一次,所以我们直接每次倍增找到最上面的没有统计的节点用树状数组+\(dfs\) 序判断是否合法就好了。

时间复杂度:\(O(n\log n)\)(默认 \(n,m\) 同级)


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#include<set>
#define lowbit(x) (x&-x)
using namespace std;
const int M=1e6+10,N=2e6+10,T=21;
struct node{
int to,next;
}a[N];
int n,m,k,last,cnt,tot,ans2;long long ans1;
int ch[M][26],len[N],fa[N],L[M],R[M],p[M],las[M];
int ls[N],v[N],pos[N],f[N][T],rfn[N],ed[N],dos[M];
set<int> ct[M];vector<int> q[M];char s[M];
set<int>::iterator it;
void Insert(int c){
int p=last,np=last=++cnt;len[np]=len[p]+1;
for(;p&&!ch[p][c];p=fa[p])ch[p][c]=np;
if(!p)fa[np]=1;
else{
int q=ch[p][c];
if(len[q]==len[p]+1)fa[np]=q;
else{
int nq=++cnt;len[nq]=len[p]+1;
memcpy(ch[nq],ch[q],sizeof(ch[nq]));
fa[nq]=fa[q];fa[q]=fa[np]=nq;
for(;p&&ch[p][c]==q;p=fa[p])ch[p][c]=nq;
}
}
return;
}
void work(int p,int l,int r){
int x=pos[r];
for(int i=T-1;i>=0;i--)
if(len[f[x][i]]>=r-l+1)x=f[x][i];
if(ct[x].count(r-l+1))return;
ct[x].insert(r-l+1);q[x].push_back(p);
return;
}
void addl(int x,int y){
a[++tot].to=y;
a[tot].next=ls[x];
ls[x]=tot;return;
}
bool cmp(int x,int y)
{return R[x]-L[x]+1<R[y]-L[y]+1;}
bool cMp(int x,int y)
{return len[x]<len[y];}
void dfs(int x){
rfn[x]=++cnt;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
f[y][0]=x;dfs(a[i].to);
}
ed[x]=cnt;return;
}
struct TreeBinary{
int t[N];
void Change(int x,int val){
while(x<=n){
t[x]+=val;
x+=lowbit(x);
}
return;
}
int Ask(int x){
int ans=0;
while(x){
ans+=t[x];
x-=lowbit(x);
}
return ans;
}
}B;
int main()
{
scanf("%d%d%d",&n,&m,&k);
scanf("%s",s+1);cnt=last=1;
for(int i=1;i<=n;i++)Insert(s[i]-'a'),pos[i]=last;
for(int i=1;i<=cnt;i++)f[i][0]=fa[i];
for(int j=1;j<T;j++)
for(int i=1;i<=cnt;i++)
f[i][j]=f[f[i][j-1]][j-1];
for(int i=1;i<=m;i++){
scanf("%d%d",&L[i],&R[i]);
work(i,L[i],R[i]);
}
for(int i=1;i<=cnt;i++)p[i]=i,ct[i].insert(len[i]);
sort(p+1,p+1+cnt,cMp);
int pnt=cnt;cnt=1;las[1]=1;
for(int i=2;i<=pnt;i++){
int x=p[i];
sort(q[x].begin(),q[x].end(),cmp);
int z=0;las[x]=las[fa[x]];
it=ct[x].begin();
while(1){
++cnt;addl(las[x],cnt);las[x]=cnt;
int W=*it;len[cnt]=*it;
while(z<q[x].size()&&R[q[x][z]]-L[q[x][z]]+1<=W)
dos[q[x][z]]=cnt,z++;
it++;if(it==ct[x].end())break;
}
}
n=cnt;cnt=0;dfs(1);v[0]=1;len[0]=-1;
for(int j=1;j<T;j++)
for(int i=1;i<=n;i++)
f[i][j]=f[f[i][j-1]][j-1];
for(int i=1;i<=m;i++){
int p=dos[i];
if(!p){printf("%lld %d\n",ans1,ans2);continue;}
B.Change(rfn[p],1);
if(v[p]){printf("%lld %d\n",ans1,ans2);continue;}
while(!v[p]){
int x=p;
for(int j=T-1;j>=0;j--)
if(!v[f[x][j]])x=f[x][j];
int w=B.Ask(ed[x])-B.Ask(rfn[x]-1);
if(w>k){
v[x]=1;
ans1+=len[x]-len[f[x][0]];
ans2=max(ans2,len[x]);
}
else break;
}
printf("%lld %d\n",ans1,ans2);
}
return 0;
}

最新文章

  1. eclipse设置快速提示符
  2. Entity framework 级联删除注意事项
  3. PHP Socket实现websocket(一)基本函数介绍
  4. 【GoLang】函数作为 类型 和 值
  5. sqlcmd
  6. 重拾java系列一java基础(4)
  7. Python修饰器的函数式编程
  8. MySQL 中随机抽样:order by rand limit 的替代方案
  9. Ubuntu 16.04 Vysor 破解 和黑屏问题解决+ 闪屏问题解决
  10. HDU 4706 Children&#39;s Day(简单模拟)
  11. 初探swift语言的学习笔记四(类对象,函数)
  12. Asp.Net BulletedList使用及详解
  13. git 由http切换成git
  14. MySQL数据库操作常用命令
  15. bit 和 byte
  16. ES6 - Map
  17. classLoader和Class.forName的区别
  18. CRM项目(一)
  19. 【原创 Hadoop&amp;Spark 动手实践 13】Spark综合案例:简易电影推荐系统
  20. jenkins shell部署

热门文章

  1. docker配置cdn-容器内可以通过域名访问
  2. HttpURLConnection 中Cookie 使用
  3. C# 使用正则表达式替换PPT中的文本(附vb.net代码)
  4. 使用栅格系统开发响应式页面——logo+nav实例
  5. 关于通用Mapper new Example使用记录
  6. 笔记:如何使用postgresql做顺序扣减库存
  7. Spring Boot 入门系列(二十四)多环境配置,3分钟搞定!
  8. JS021. 拦截事件的显式处理与默认动作(Web API: event.preventDefault)
  9. Element UI:DatePicker的终止日期与起始日期关联
  10. 浅谈KMP模式匹配算法