思路

在后缀树上进行一些操作就好了

后缀树上LCA的maxlen就是两个后缀的LCP的长度了

然后统计每个点作为LCA的次数和最大值、次大值、最小值、次小值

然后就做完了

代码

#include <cstdio>
#include <algorithm>
#include <cstring>
#include <queue>
#define int long long
using namespace std;
int val[301000*2],n;
char s[301000*2];
namespace SAM{
int maxlen[301000*2],suflink[301000*2],trans[301000*2][26],endpos[301000*2],maxnum[301000*2],semaxnum[301000*2],minnum[301000*2],seminnum[301000*2],Nodecnt=1,in[301000*2],num[301000*2],maxval[301000*2];
int fir[301000*2],v[301000*2],nxt[301000*2],cnt=0;
void addedge(int ui,int vi){
++cnt;
v[cnt]=vi;
nxt[cnt]=fir[ui];
fir[ui]=cnt;
}
int New_state(int _suflink,int *_trans,int _maxlen){
++Nodecnt;
suflink[Nodecnt]=_suflink;
if(_trans)
for(int i=0;i<26;i++)
trans[Nodecnt][i]=_trans[i];
maxlen[Nodecnt]=_maxlen;
return Nodecnt;
}
int add_len(int u,int c,int inq){
int z=New_state(0,NULL,maxlen[u]+1);
endpos[z]=1;
maxnum[z]=minnum[z]=inq;
semaxnum[z]=-0x3f3f3f3f;
seminnum[z]=0x3f3f3f3f;
while(u&&trans[u][c]==0){
trans[u][c]=z;
u=suflink[u];
}
if(!u){
suflink[z]=1;
return z;
}
int v=trans[u][c];
if(maxlen[v]==maxlen[u]+1){
suflink[z]=v;
return z;
}
int y=New_state(suflink[v],trans[v],maxlen[u]+1);
endpos[y]=0;
maxnum[y]=semaxnum[y]=-0x3f3f3f3f;
minnum[y]=seminnum[y]=0x3f3f3f3f;
suflink[z]=suflink[v]=y;
while(u&&trans[u][c]==v){
trans[u][c]=y;
u=suflink[u];
}
return z;
}
void insert(char *s,int len){
reverse(s+1,s+len+1);
reverse(val+1,val+len+1);
int last=1;
for(int i=1;i<=len;i++)
last=add_len(last,s[i]-'a',val[i]);
}
void build(void){
memset(num,0,sizeof(num));
memset(maxval,-0x3f,sizeof(maxval));
for(int i=1;i<=Nodecnt;i++)
addedge(suflink[i],i);
}
void dfs(int u){
for(int i=fir[u];i;i=nxt[i]){
dfs(v[i]);
num[maxlen[u]]+=endpos[u]*endpos[v[i]];
endpos[u]+=endpos[v[i]];
if(maxnum[v[i]]>=maxnum[u]){
if(max(maxnum[u],semaxnum[v[i]])>semaxnum[u])
semaxnum[u]=max(maxnum[u],semaxnum[v[i]]);
maxnum[u]=maxnum[v[i]];
}
else if(maxnum[v[i]]>semaxnum[u]){
semaxnum[u]=maxnum[v[i]];
}
if(minnum[v[i]]<=minnum[u]){
if(min(seminnum[v[i]],minnum[u])<seminnum[u])
seminnum[u]=min(seminnum[v[i]],minnum[u]);
minnum[u]=minnum[v[i]];
}
else if(minnum[v[i]]<seminnum[u]){
seminnum[u]=minnum[v[i]];
}
}
if(endpos[u]>1)
maxval[maxlen[u]]=max(maxval[maxlen[u]],max(maxnum[u]*semaxnum[u],minnum[u]*seminnum[u]));
}
};
signed main(){
scanf("%lld",&n);
scanf("%s",s+1);
for(int i=1;i<=n;i++)
scanf("%lld",&val[i]);
SAM::insert(s,n);
SAM::build();
SAM::dfs(1);
for(int i=n;i>=0;i--){
SAM::maxval[i]=max(SAM::maxval[i],SAM::maxval[i+1]);
SAM::num[i]+=SAM::num[i+1];
}
for(int i=0;i<n;i++)
printf("%lld %lld\n",SAM::num[i],(SAM::num[i]==0)?0:SAM::maxval[i]);
return 0;
}

P2178 [NOI2015]品酒大会的更多相关文章

  1. 洛谷 P2178 [NOI2015]品酒大会 解题报告

    P2178 [NOI2015]品酒大会 题目描述 一年一度的"幻影阁夏日品酒大会"隆重开幕了.大会包含品尝和趣味挑战 两个环节,分别向优胜者颁发"首席品酒家"和 ...

  2. 洛谷P2178 [NOI2015]品酒大会 后缀数组+单调栈

    P2178 [NOI2015]品酒大会 题目链接 https://www.luogu.org/problemnew/show/P2178 题目描述 一年一度的"幻影阁夏日品酒大会" ...

  3. 洛谷P2178 [NOI2015]品酒大会

    题目描述 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战 两个环节,分别向优胜者颁发“首席品酒家”和“首席猎手”两个奖项,吸引了众多品酒师参加. 在大会的晚餐上,调酒师 Rainb ...

  4. 洛谷P2178 [NOI2015]品酒大会(后缀自动机 线段树)

    题意 题目链接 Sol 说一个后缀自动机+线段树的无脑做法 首先建出SAM,然后对parent树进行dp,维护最大次大值,最小次小值 显然一个串能更新答案的区间是\([len_{fa_{x}} + 1 ...

  5. BZOJ 4199: [Noi2015]品酒大会 [后缀数组 带权并查集]

    4199: [Noi2015]品酒大会 UOJ:http://uoj.ac/problem/131 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品 ...

  6. [UOJ#131][BZOJ4199][NOI2015]品酒大会 后缀数组 + 并查集

    [UOJ#131][BZOJ4199][NOI2015]品酒大会 试题描述 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品酒家”和“首席猎手”两个 ...

  7. 【BZOJ4199】[Noi2015]品酒大会 后缀数组+并查集

    [BZOJ4199][Noi2015]品酒大会 题面:http://www.lydsy.com/JudgeOnline/wttl/thread.php?tid=2144 题解:听说能用SAM?SA默默 ...

  8. [UOJ#131][BZOJ4199][NOI2015]品酒大会

    [UOJ#131][BZOJ4199][NOI2015]品酒大会 试题描述 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品酒家”和“首席猎手”两个 ...

  9. BZOJ_4199_[Noi2015]品酒大会_后缀自动机

    BZOJ_4199_[Noi2015]品酒大会_后缀自动机 Description 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品 酒家”和“首席 ...

随机推荐

  1. LeetCode 217 Contains Duplicate 解题报告

    题目要求 Given an array of integers, find if the array contains any duplicates. Your function should ret ...

  2. python 日常代码 tips

    1. 官方示例很多情况是用的列表的形式,bokeh本身不是基于pandas构建的可视化工具,所以它基本上是用的python自己的数据结构字典.列表:我们做数据分析肯定是基于pandas,以上就是做了一 ...

  3. mysql (六)

    约束: 非空约束: 创建表时添加: create table stu( id  int , name  varchar(20) not null ) 创建完表之后 alter table stu  m ...

  4. callback function(回调函数) - 术语

    回调函数(CallBack Function)的定义: [todo] 下面是callback的一个场景,linux内核.LwIP里也有使用callback funcs.

  5. 小程序 滚动wx.pageScrollTo

    API:https://developers.weixin.qq.com/miniprogram/dev/api/wx.pageScrollTo.html wx.pageScrollTo 在小程序的开 ...

  6. Extjs6 grid 导出excel功能类,支持renderer

    /* grid 导出excel扩展(纯客户端,提交到后台再导的可以自己改改代码也在) 参考自 https://blog.csdn.net/tianxiaode/article/details/4596 ...

  7. pandas处理时间序列(4): 移动窗口函数

    六.移动窗口函数 移动窗口和指数加权函数类别如↓: rolling_mean 移动窗口的均值 pandas.rolling_mean(arg, window, min_periods=None, fr ...

  8. libpng warning: iCCP: known incorrect sRGB profile

    参考  http://www.cocos2d-x.org/forums/6/topics/49093 解决 I got the following warnings in console when r ...

  9. CarbonData-1:common

    最近公司需要对CarbonData进一步应用,或许封装进产品,或许是为了解析CarbonData元数据,于是开始预研CarbonData,下面将保持每天一篇以上的阅读CarbonData源码博客,由于 ...

  10. Ubuntu 自动选择最快的镜像源

    通常情况下阿里云的镜像源(http://mirrors.aliyun.com/ubuntu/)用得比较多 但是也不排除因网络环境导致的某个镜像源访问慢问题 那么就可以配置为自动选择镜像源进行更新 修改 ...