[bzoj4199][Noi2015]品酒大会_后缀自动机_后缀树_树形dp
品酒大会 bzoj-4199 Noi-2015
题目大意:给定一个字符串,如果其两个子串的前$r$个字符相等,那么称这两个子串的开头两个位置$r$相似。如果两个位置勾兑在一起那么美味度为两个位置的乘积。
注释:$1\le length \le 3\cdot 10^5$。
想法:我们先建立后缀自动机。
然后求出后缀树。
显然如果在后缀树上一个节点是另一个节点的祖先,那么这个节点代表的所有字符串一定是另一个节点代表的所有字符串的后缀。
唔....
这个时候我们发现不太对,于是就对反串建好了。
建立出后缀树时候,我们树形$dp$就好了啊。
然后我们求$f[i]$表示恰好$i$相似的方案数。
那么$f[i]$就是$dis$为$i$的子树两两儿子$size$和。
之后$g[i]$就是$dis$为$i$的子树中两个最大的点的乘积。
有个小问题就是权值可能为负所以还要维护最小值和次小值。
最后前缀加和前缀取$max$统计一下即可。
代码:
#include <bits/stdc++.h>
#define N 600010
#define MAX (0x7f7f7f7f)
#define MIN (0x80808080)
using namespace std; typedef long long ll;
int w[N],a[N][26],fa[N],dis[N],lst=1,cnt=1,head[N],to[N],nxt[N],tot,size[N],mx[N],mn[N];
ll ans1[N],ans2[N]; char str[N];
inline void add(int x,int y) {to[++tot]=y; nxt[tot]=head[x]; head[x]=tot;}
void update(int c,int v)
{
int p=lst,np=lst=++cnt;
dis[np]=dis[p]+1; mx[np]=mn[np]=v; size[np]=1;
while(p&&!a[p][c]) a[p][c]=np,p=fa[p];
if(!p) {fa[np]=1; return;}
int q=a[p][c];
if(dis[q]==dis[p]+1) {fa[np]=q; return;}
int nq=++cnt;
memcpy(a[nq],a[q],sizeof a[q]); dis[nq]=dis[p]+1;
fa[nq]=fa[q]; fa[np]=fa[q]=nq;
while(p&&a[p][c]==q) a[p][c]=nq,p=fa[p];
}
void dfs(int x)
{
int p=dis[x]; for(int i=head[x];i;i=nxt[i])
{
dfs(to[i]);
ans1[p]+=(ll)size[x]*size[to[i]];
if(mx[x]!=MIN) ans2[p]=max(ans2[p],(ll)mx[x]*mx[to[i]]);
if(mn[x]!=MAX) ans2[p]=max(ans2[p],(ll)mn[x]*mn[to[i]]);
size[x]+=size[to[i]];
mx[x]=max(mx[x],mx[to[i]]); mn[x]=min(mn[x],mn[to[i]]);
}
}
int main()
{
int n; scanf("%d%s",&n,str+1);
memset(mx,0x80,sizeof mx); memset(mn,0x7f,sizeof mn); memset(ans2,0x80,sizeof ans2);
for(int i=1;i<=n;i++) scanf("%d",&w[i]);
for(int i=n;i;i--) update(str[i]-'a',w[i]);
for(int i=2;i<=cnt;i++) add(fa[i],i);
dfs(1); for(int i=n-2;~i;i--) ans1[i]+=ans1[i+1],ans2[i]=max(ans2[i],ans2[i+1]);
for(int i=0;i<n;i++) printf("%lld %lld\n",ans1[i],ans1[i]?ans2[i]:0);
return 0;
}
小结:后缀自动机求出的后缀树还是非常好用的。
[bzoj4199][Noi2015]品酒大会_后缀自动机_后缀树_树形dp的更多相关文章
- [UOJ#131][BZOJ4199][NOI2015]品酒大会 后缀数组 + 并查集
[UOJ#131][BZOJ4199][NOI2015]品酒大会 试题描述 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品酒家”和“首席猎手”两个 ...
- [UOJ#131][BZOJ4199][NOI2015]品酒大会
[UOJ#131][BZOJ4199][NOI2015]品酒大会 试题描述 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品酒家”和“首席猎手”两个 ...
- [BZOJ4199][Noi2015]品酒大会 树形DP+后缀自动机
由于要找后缀的前缀,所以先用反串建立SAM. link边组成了后缀树. 两个子串的最长公共前缀是LCA的step 树形dp即可. #include<iostream> #include&l ...
- bzoj4199: [Noi2015]品酒大会(后缀数组)
题目描述 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战 两个环节,分别向优胜者颁发“首席品酒家”和“首席猎手”两个奖项,吸引了众多品酒师参加. 在大会的晚餐上,调酒师 Rainb ...
- BZOJ4199 [Noi2015]品酒大会 【后缀数组 + 单调栈 + ST表】
题目 一年一度的"幻影阁夏日品酒大会"隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发"首席品 酒家"和"首席猎手"两个奖项,吸 ...
- bzoj4199: [Noi2015]品酒大会 (并查集 && 后缀数组)
据说用后缀自动机 + dp也能做 然而并不会 后缀数组的做法呢 就是先建个后缀数组,求出height值,此时如果直接找,复杂度是n ^ 2的,肯定会超时. 但是height大的值是不会对小的产生影响的 ...
- [BZOJ4199][NOI2015]品酒大会
#131. [NOI2015]品酒大会 统计 描述 提交 自定义测试 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品酒家”和“首席猎手”两个奖项, ...
- 并不对劲的bzoj4199: [Noi2015]品酒大会
传送门-> 又称普及大会. 这题没什么好说的……后缀自动机裸题……并不对劲的人太菜了,之前照着标程逐行比对才过了这道题,前几天刚刚把这题一遍写对…… 这题的输出和某两点相同后缀的长度有关,那么把 ...
- 2019.02.28 bzoj4199: [Noi2015]品酒大会(sam+线段树)
传送门 题意:给一个串,每个位置有一个权值,当S[s...s+len−1]=S[t...t+len−1]&&S[s...s+len]̸=S[t..t+len]S[s...s+len-1 ...
随机推荐
- ASP.NET Web API FilterAttribute假想
偶然的测试发现API FilterAttribute没用引用只会初始化一次 比如: 如果是 Global Action Filter, 则全局只会初始化一次 针对于不同的Controller级别的Ac ...
- Android中单选框RadioButton的基本用法
总结一下设置图标的三种方式: (1)button属性:主要用于图标大小要求不高,间隔要求也不高的场合. (2)background属性:主要用于能够以较大空间显示图标的场合. (3)drawableL ...
- php bz2扩展安装
php bz2扩展安装 2017年09月22日 14:14:36 Cookie_1030 阅读数:1781 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn ...
- Python3基础教程(十六)—— 迭代器、生成器、装饰器
在这个实验里我们学习迭代器.生成器.装饰器有关知识. 这几个概念是 Python 中不容易理解透彻的概念,务必把所有的实验代码都完整的输入并理解清楚其中每一行的意思. 迭代器 Python 迭代器(I ...
- Python3简明教程(二)—— 变量和数据类型
关键字和标识符 下列的标识符是Python3的关键字,并且不能用于通常的标识符.关键字必须严格按照下面的拼写: False def if raise None del import return Tr ...
- Visual Studio中Radio Button组绑定变量方法(DDX_Radio方法)
需求描述:Visual Studio 创建的界面程序中又许多 Radio Button,希望这些所有的Radio Button统一绑定到一个变量上,这个变量一旦改变,Radio Button的选中状态 ...
- U盘制作安装盘后容量不能恢复的解决方案
diskpartlist diskselect disk 0/1 --看具体U盘是0还是1clean
- [题解] cogs 1669 神秘的咒语
http://cogs.pro:8080/cogs/problem/problem.php?pid=1669 "The Real Incantation is Their Common In ...
- xenserver tools 安装
mkdir -p /mnt/xtools mount /dev/cdrom /mnt/xtools cd /mnt/xtools/Linux/ ./install.sh -n init 6
- Android开发——后台获取用户点击位置坐标(可获取用户支付宝密码)
1. getevent命令 我们首先是根据adb shell getevent命令获取到被点击位置的信息. 这里要说明的是,不同的手机手机获得的点击输出是不一样的.以我的真机为例,输出如下 本文原创, ...