思路

广义SAM的题目,先全部插入,然后每个字符串在SAM上匹配,如果发现当前sz小于k(就是前缀不满足条件),就跳fail(找前缀的后缀,就是找子串)到满足条件为止,然后一个满足条件的节点,它的所有parent Tree上的祖先也必定满足条件,所以一个满足条件的节点的贡献就是它的maxlen

ps:貌似求sz除了我这种,还可以set启发式合并

代码

#include <cstdio>
#include <algorithm>
#include <cstring>
#include <string>
#include <iostream>
#include <queue>
using namespace std;
int maxlen[201000],minlen[201000],trans[201000][26],suflink[201000],ans[201000],sz[201000],last[201000],endpos[201000],Nodecnt,n;
string s[101000];
int New_state(int _maxlen,int _minlen,int *_trans,int _sz,int _last,int _suflink){
++Nodecnt;
maxlen[Nodecnt]=_maxlen;
minlen[Nodecnt]=_minlen;
if(_trans)
for(int i=0;i<26;i++)
trans[Nodecnt][i]=_trans[i];
sz[Nodecnt]=_sz;
last[Nodecnt]=_last;
suflink[Nodecnt]=_suflink;
return Nodecnt;
}
void update(int u,int x){
while(u&&last[u]!=x){
last[u]=x;
sz[u]++;
u=suflink[u];
}
}
int add_len(int u,int c,int inq){
if(trans[u][c]){
int v=trans[u][c];
if(maxlen[v]==maxlen[u]+1){
update(v,inq);
endpos[v]++;
return v;
}
int y=New_state(maxlen[u]+1,0,trans[v],sz[v],last[v],suflink[v]);
suflink[v]=y;
minlen[v]=maxlen[y]+1;
while(u&&trans[u][c]==v){
trans[u][c]=y;
u=suflink[u];
}
minlen[y]=maxlen[suflink[y]]+1;
update(y,inq);
return y;
}
else{
int z=New_state(maxlen[u]+1,0,NULL,0,0,0);
endpos[z]=1;
while(u&&trans[u][c]==0){
trans[u][c]=z;
u=suflink[u];
}
if(!u){
suflink[z]=1;
minlen[z]=1;
update(z,inq);
return z;
}
int v=trans[u][c];
if(maxlen[v]==maxlen[u]+1){
suflink[z]=v;
minlen[z]=maxlen[v]+1;
update(z,inq);
return z;
}
int y=New_state(maxlen[u]+1,0,trans[v],sz[v],last[v],suflink[v]);
suflink[v]=suflink[z]=y;
minlen[v]=minlen[z]=maxlen[y]+1;
while(u&&(trans[u][c]==v)){
trans[u][c]=y;
u=suflink[u];
}
minlen[y]=maxlen[suflink[y]]+1;
update(z,inq);
return z;
}
}
int k,in[200100];
queue<int> q;
int main(){
freopen("test.in","r",stdin);
freopen("test.out","w",stdout);
Nodecnt=1;
scanf("%d %d",&n,&k);
for(int i=1;i<=n;i++){
// scanf("%s",s+1);
cin>>s[i];
int last=1,len=s[i].length();
// cout<<len<<endl;
for(int j=0;j<len;j++)
last=add_len(last,s[i][j]-'a',i);
}
for(int i=1;i<=n;i++){
int lent=s[i].length(),p=1;
// cout<<"!"<<lent<<endl;
int ans=0;
for(int j=0;j<lent;j++){
p=trans[p][s[i][j]-'a'];
while(sz[p]<k)
p=suflink[p];
if(p==0){
p=1;
continue;
}
ans+=(maxlen[p]);
}
printf("%d ",ans);
}
return 0;
}

最新文章

  1. (47) odoo详细操作手册
  2. File存对象--android 的File存储到SD卡();
  3. jq 构造函数,然后再表单提交过程中对数据进行修改
  4. Xamarin Android 真机调试时闪退
  5. 在linux下配置静态IP
  6. 获取bing每日图片
  7. android Camera 录像时旋转角度
  8. 洛谷P1993 小 K 的农场
  9. MySQL初级培训
  10. socket通信技术介绍
  11. html 细线表格
  12. Java Socket通信以及可能出现的问题解决
  13. Docker Image管理学习笔记,ZT
  14. Redis学习系列七分布式锁
  15. 封装||property
  16. HTML Tags
  17. c++ sizeof,strlen, length
  18. 基于FPGA的I2C读写EEPROM
  19. POJ 1117 Pairs of Integers
  20. JAVA学习(七)__Spring的@Autowired注入规则

热门文章

  1. oracle04 约束,索引
  2. pandas(三)
  3. 2017(5)软件架构设计,web系统的架构设计,数据库系统,分布式数据库
  4. JavaScript实现预览本地上传图片
  5. T-SQL语言基础(2)之SQL Server体系结构
  6. windows下复制文件报错“文件名对目标文件夹可能过长 。您可以缩短文件名并重试,或者......”
  7. sping Bean 的生命周期是如何被管理
  8. Java之.jdk卸载-Linux
  9. Pyenv部署
  10. C# 链表反转