loj6198谢特 后缀数组+并查集+Trie
先把问题放在后缀数组上考虑
已知两个数组a b,求min(a[i],...,a[j])+(b[i]^b[j])的最大值
套路题
初始每个点都是一个小连通块
把a按从大到小的顺序加入,计算当前加入边作为min的贡献:
每次加入会把两个连通块联通,答案就是两边连通块各出一个数能得到的异或和最大值
我:这不是线性基吗
miaom:mdzz,只能有两个数
我:蛤,好难啊,怎么做啊
miaom:Trie啊
我:哦
没了
#include <bits/stdc++.h>
#define N 500001
#define MAX 18
using namespace std;
int NODE,n;char ch;
int a[N],b[N],c[N*MAX][],size[N],rt[N],lef[N];
pair<int,int> so[N];
int X[N],Y[N],SA[N],Rk[N],Ht[N],v[N];
void GetSA(int a[],int n,int m=)
{
int *x=X,*y=Y,i,p,j;
memset(v,,sizeof v);
for(i=;i<=n;i++)v[x[i]=a[i]]++;
for(i=;i<=m;i++)v[i]+=v[i-];
for(i=n;i>=;i--)SA[v[x[i]]--]=i;
for(i=;i<=n;i<<=,m=p)
{
p=;
memset(v,,sizeof v);
for(j=n-i+;j<=n;j++)y[++p]=j;
for(j=;j<=n;j++)if(SA[j]>i)y[++p]=SA[j]-i;
for(j=;j<=n;j++)v[x[y[j]]]++;
for(j=;j<=m;j++)v[j]+=v[j-];
for(j=n;j>=;j--)SA[v[x[y[j]]]--]=y[j];
swap(x,y);
p=;x[SA[]]=;
for(j=;j<=n;j++)
if(y[SA[j-]]==y[SA[j]]&&y[SA[j-]+i]==y[SA[j]+i]) x[SA[j]]=p;
else x[SA[j]]=++p;
if(p>=n)break;
}
}
void GetHt(int a[],int n)
{
int k=;
for(int i=;i<=n;i++)Rk[SA[i]]=i;
for(int i=;i<=n;i++)
{
if(k)k--;
int j=SA[Rk[i]-];
while(i+k<=n&&j+k<=n&&a[i+k]==a[j+k])k++;
Ht[Rk[i]]=k;
}
}
int build(int x)
{
int rt=++NODE,now=rt;
for(int i=MAX;i>=;i--)
c[now][(x>>i)&]=++NODE,now=NODE;
return rt;
}
int que(int x,int y,int z=)
{
int best=z;
if(c[x][])
if(c[y][]) best=que(c[x][],c[y][],z*+);
else best=que(c[x][],c[y][],z*);
if(c[x][])
if(c[y][]) best=max(best,que(c[x][],c[y][],z*+));
else best=max(best,que(c[x][],c[y][],z*));
return best;
}
void merge(int x,int y)
{
if(c[x][])
if(c[y][]) merge(c[x][],c[y][]);
else c[y][]=c[x][];
if(c[x][])
if(c[y][]) merge(c[x][],c[y][]);
else c[y][]=c[x][];
}
int getfa(int x)
{
if(lef[x]==x) return x;
else return lef[x]=getfa(lef[x]);
}
int main()
{
scanf("%d",&n);
for(ch=getchar();!isalpha(ch);ch=getchar());
for(int i=;i<=n;i++,ch=getchar())
a[i]=ch;
for(int i=;i<=n;i++)
scanf("%d",&b[i]);
GetSA(a,n);
GetHt(a,n);
for(int i=;i<=n;i++)
size[i]=,lef[i]=i,rt[i]=build(b[SA[i]]);
for(int i=;i<=n;i++)
so[i-]=make_pair(n-Ht[i],i);
sort(so+,so+n);
int ret=;
for(int i=;i<n;i++)
{
int x=so[i].second,y=x-,z=n-so[i].first;
x=getfa(x);y=getfa(y);
if(size[x]<size[y]) swap(x,y);
int bes=que(rt[y],rt[x])+z;
if(bes>ret)
ret=bes;
merge(rt[y],rt[x]);
size[x]+=size[y];
lef[y]=x;
}
printf("%d\n",ret);
return ;
}
loj6198谢特 后缀数组+并查集+Trie的更多相关文章
- BZOJ 4566 JZYZOJ 1547 [haoi2016T5]找相同子串 后缀数组 并查集
http://172.20.6.3/Problem_Show.asp?id=1547 http://www.lydsy.com/JudgeOnline/problem.php?id=4566 单纯后缀 ...
- NOI 2015 品酒大会 (后缀数组+并查集)
题目大意:略 40分暴力还是很好写的,差分再跑个后缀和 和 后缀最大值就行了 一种正解是后缀数组+并查集 但据说还有后缀数组+单调栈的高端操作蒟蒻的我当然不会 后缀数组求出height,然后从大到小排 ...
- [UOJ#131][BZOJ4199][NOI2015]品酒大会 后缀数组 + 并查集
[UOJ#131][BZOJ4199][NOI2015]品酒大会 试题描述 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品酒家”和“首席猎手”两个 ...
- Uva 12361 File Retrieval 后缀数组+并查集
题意:有F个单词,1 <= F <=60 , 长度<=10^4, 每次可以输入一个字符串,所有包含该字串的单词会形成一个集合. 问最多能形成多少个不同的集合.集合不能为空. 分析:用 ...
- BZOJ 4199: [Noi2015]品酒大会( 后缀数组 + 并查集 )
求出后缀数组后, 对height排序, 从大到小来处理(r相似必定是0~r-1相似), 并查集维护. 复杂度O(NlogN + Nalpha(N)) ------------------------- ...
- 【学术篇】NOI2015 品酒大会 后缀数组+并查集
省选前大致是刷不了几道题了... 所以就找一些裸一点的题目练练板子算了= = 然而这题一点都不裸, 也并不怎么好写... 于是就浪费了将近一下午的时间... 然而还不是因为后缀数组板子不熟= = 首先 ...
- 4199. [NOI2015]品酒大会【后缀数组+并查集】
Description 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品 酒家”和“首席猎手”两个奖项,吸引了众多品酒师参加.在大会的晚餐上,调酒师 ...
- 【BZOJ4199】[Noi2015]品酒大会 后缀数组+并查集
[BZOJ4199][Noi2015]品酒大会 题面:http://www.lydsy.com/JudgeOnline/wttl/thread.php?tid=2144 题解:听说能用SAM?SA默默 ...
- CF 452E. Three strings(后缀数组+并查集)
传送门 解题思路 感觉这种题都是套路之类的??首先把三个串并成一个,中间插入一些奇怪的字符,然后跑遍\(SA\).考虑按照\(height\)分组计算,就是每个\(height\)只在最高位计算一次, ...
随机推荐
- sublime 相关配置和快捷键
1.安装package control 点击sublime的菜单栏 view->show console :现在打开了控制台, 这个控制台有上下两栏, 上面一栏会实时显示sublime执行了什 ...
- ffmpeg给视频加文字水印
ffmpeg -i dd2800.mp4 -vf "drawtext=fontfile=Arial.ttf: text='Hu':x=100:y=10:fontsize=24:fontcol ...
- 一个关于前端页面的小标签<tbody>
我们有时候希望将表格的内容分为多个模块,这时候就可以使用<tbody>标签,它是<table>的字标签,是<tr>的父标签,可以使用它达到一种设置样式的结果.
- 【HDU 6126】Give out candies 最小割
题意 有$n$个小朋友,给每个人分$1~m$个糖果,有k个限制 限制形如$(x,y,z)$ 表示第$x$个人分到的糖数减去第$y$个人分到的糖数不大于$z$,给第$i$个人$j$颗糖获 ...
- 「LuoguP4995」「洛谷11月月赛」 跳跳!(贪心
题目描述 你是一只小跳蛙,你特别擅长在各种地方跳来跳去. 这一天,你和朋友小 F 一起出去玩耍的时候,遇到了一堆高矮不同的石头,其中第 ii 块的石头高度为 h_ihi,地面的高度是 h_0 = 0 ...
- bzoj4555: 求和sum 快速傅立叶变换
题目大意 给定\(S(n,m)\)表示第二类斯特林数,定义函数\(f(n)\) \[f(n) = \sum_{i=0}^n\sum_{j=0}^iS(i,j)*2^j*(j!)\] 给定正整数\(n, ...
- python+ mysql存储二进制流的方式
很多时候我们为了管理方便会把依稀很小的图片存入数据库,有人可能会想这样会不会对数据库造成很大的压力,其实大家可以不用担心,因为我说过了,是存储一些很小的图片,几K的,没有问题的! 再者,在这里我们是想 ...
- SQL Server 将查询的结果生成insert语句
1.将查询的结果插入到一张新表(yangTest 表系统会自动生成)select Area, District, RoadName, StationName, PathDirection, Stati ...
- hibernate学习五 Hibernate补充
1 MiddleGenIDE可以生成映射类和映射文件. 2
- rmmod: chdir(/lib/modules): No such file or directory
内核版本:linux3.4.20 交叉编译器:arm-linux-gcc 4.3.3 busybox : busybox 1.20 问题: 使用rmmod会出现 rmmod : chdir(/lib ...