【loj2033】生成魔咒
Solution
这题。。虽然说好像也是sam的裸题不过既然在智力康复那就强制后缀数组吧qwq
(晚点再用sam写一次qwq)
首先如果是要求本质不同的串的数量的话,如果说只用求一次,那么我们直接跑出这个串的\(Sa\)和\(height\),然后直接对于每一位\(i\)将\(ans+=(n-sa[i]+1)-height[i]\),具体的话可以看这里Portal-->
然后现在的问题是。。每加入一个字符我们都需要这么求一次
那么我们可以考虑将这个串反过来,这样往末端加字符的操作就变成了往前段加字符,也就是变成了多加一个后缀,这样我们就可以比较好处理这个问题了
我们先对原串的反串求出\(Sa\)和\(height\),然后用一个set或者。。额其实链表也是可以的来维护当前有哪些后缀(存\(rk\)值就好了),顺序按照这些后缀的\(rk\)值来排,每次我们找到插入当前这个后缀的位置,记加入这个后缀之后,前面的那个位置的\(rk\)值为\(pre\),后面那个位置的\(rk\)值为\(nxt\),那么我们就将原来的贡献(也就是\((n-sa[nxt]+1)-lcp(pre,nxt)\))减去,然后将新的贡献也就是当前后缀和\(pre\)的贡献、当前后缀和\(nxt\)的贡献加上,一直这么操作就好了
需要注意的是,这里要用long long,并且一开始的时候\(set\)中应该有一个\(0\),这样才能将第一个后缀的贡献算进去,以及因为数字\(x\)的范围有点吓人所以我们需要离散化一下
代码大概长这个样子
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#define ll long long
using namespace std;
const int N=1e5+10,TOP=20,inf=2147483647;
int s[N],lis[N];
ll ans[N];
int n;
namespace Sa{/*{{{*/
set<int> rec;
set<int>::iterator it,pre,nxt;
int a[N],b[N],c[N],sa[N],height[N],rk[N];
int mn[N][TOP+1];
int tot,mx,n;
bool cmp(int x,int y,int len,int *r)
{return r[x]==r[y]&&r[x+len]==r[y+len];}
void sort(int n){
for (int i=0;i<=mx;++i) c[i]=0;
for (int i=1;i<=n;++i) ++c[a[b[i]]];
for (int i=1;i<=mx;++i) c[i]+=c[i-1];
for (int i=n;i>=1;--i) sa[c[a[b[i]]]--]=b[i];
}
void get_sa(int _n){
n=_n;
int cnt=0;
mx=0;
for (int i=1;i<=n;++i) a[i]=s[i],b[i]=i,mx=max(a[i],mx);
sort(n);
for (int len=1;cnt<n;len<<=1){
cnt=0;
for (int i=n-len+1;i<=n;++i) b[++cnt]=i;
for (int i=1;i<=n;++i)
if (sa[i]>len)
b[++cnt]=sa[i]-len;
sort(n);
swap(a,b);
cnt=1; a[sa[1]]=1;
for (int i=2;i<=n;a[sa[i++]]=cnt)
if (!cmp(sa[i],sa[i-1],len,b)) ++cnt;
mx=cnt;
}
}
void rmq(){
for (int i=1;i<=n;++i) mn[i][0]=height[i];
for (int j=1;j<=TOP;++j)
for (int i=n-(1<<j)+1;i>=1;--i)
mn[i][j]=min(mn[i][j-1],mn[i+(1<<j-1)][j-1]);
}
void get_height(){
for (int i=1;i<=n;++i) rk[sa[i]]=i;
int k=0;
for (int i=1;i<=n;++i){
if (k) --k;
while (s[i+k]==s[sa[rk[i]-1]+k]) ++k;
height[rk[i]]=k;
}
rmq();
}
int lcp(int x,int y){//x,y are ranks
if (x==y) return n-sa[x]+1;
if (x>y) swap(x,y);
++x;
int len=y-x+1,lg=(int)(log(1.0*len)/log(2.0));
return min(mn[x][lg],mn[y-(1<<lg)+1][lg]);
}
int get_val(int l,int r){
if (l>r) swap(l,r);
return (n-sa[r]+1)-lcp(l,r);
}
int calc(){
int ret=0;
for (int i=1;i<=n;++i)
ret+=(n-sa[i]+1)-height[i];
return ret;
}
void print(){
set<int>::iterator tmp;
for (tmp=rec.begin();tmp!=rec.end(); ++tmp)
printf("%d ",*tmp);
printf("\n");
}
void solve(int n){
rec.clear();
rec.insert(0); rec.insert(inf); ans[n+1]=0;
//print();
for (int i=n;i>=1;--i){
it=rec.insert(rk[i]).first; pre=it; nxt=it;
--pre; ++nxt;
ans[i]=ans[i+1];
if (*nxt!=inf){
ans[i]-=get_val(*pre,*nxt);
ans[i]+=get_val(*it,*nxt);
}
ans[i]+=get_val(*pre,*it);
//print();
}
}
}/*}}}*/
void prework(){
sort(lis+1,lis+1+n);
lis[0]=unique(lis+1,lis+1+n)-lis-1;
for (int i=1;i<=n;++i) s[i]=lower_bound(lis+1,lis+1+lis[0],s[i])-lis;
}
int main(){
#ifndef ONLINE_JUDGE
freopen("a.in","r",stdin);
#endif
scanf("%d",&n);
for (int i=1;i<=n;++i) scanf("%d",s+i),lis[i]=s[i];
prework();
reverse(s+1,s+1+n);
Sa::get_sa(n);
Sa::get_height();
Sa::solve(n);
for (int i=n;i>=1;--i) printf("%lld\n",ans[i]);
}
【loj2033】生成魔咒的更多相关文章
- BZOJ4516: [Sdoi2016]生成魔咒 后缀自动机
#include<iostream> #include<cstdio> #include<cstring> #include<queue> #inclu ...
- BZOJ 4516: [Sdoi2016]生成魔咒 [后缀自动机]
4516: [Sdoi2016]生成魔咒 题意:询问一个字符串每个前缀有多少不同的子串 做了一下SDOI2016R1D2,题好水啊随便AK 强行开map上SAM 每个状态的贡献就是\(Max(s)-M ...
- [SDOI2016]生成魔咒
题目描述 魔咒串由许多魔咒字符组成,魔咒字符可以用数字表示.例如可以将魔咒字符 1.2 拼凑起来形成一个魔咒串 [1,2]. 一个魔咒串 S 的非空字串被称为魔咒串 S 的生成魔咒. 例如 S=[1, ...
- BZOJ_4516_[Sdoi2016]生成魔咒_后缀数组+ST表+splay
BZOJ_4516_[Sdoi2016]生成魔咒_后缀数组+ST表+splay Description 魔咒串由许多魔咒字符组成,魔咒字符可以用数字表示.例如可以将魔咒字符 1.2 拼凑起来形成一个魔 ...
- [BZOJ 4516] [SDOI 2016] 生成魔咒
Description 魔咒串由许多魔咒字符组成,魔咒字符可以用数字表示.例如可以将魔咒字符 1.2 拼凑起来形成一个魔咒串 [1,2]. 一个魔咒串 S 的非空字串被称为魔咒串 S 的生成魔咒. 例 ...
- P4070 [SDOI2016]生成魔咒
题目地址:P4070 [SDOI2016]生成魔咒 相信看到题目之后很多人跟我的思路是一样的-- 肯定要用 SA(P3809 [模板]后缀排序) 肯定要会求本质不同的子串个数(P2408 不同子串个数 ...
- bzoj4516 / P4070 [SDOI2016]生成魔咒
P4070 [SDOI2016]生成魔咒 后缀自动机 每插入一个字符,对答案的贡献为$len[last]-len[fa[last]]$ 插入字符范围过大,所以使用$map$存储. (去掉第35行就是裸 ...
- 【LG4070】[SDOI2016]生成魔咒
[LG4070][SDOI2016]生成魔咒 题面 洛谷 题解 如果我们不用在线输的话,那么答案就是对于所有状态\(i\) \[ \sum (i.len-i.fa.len) \] 现在我们需要在线询问 ...
- 洛谷 P4070 [SDOI2016]生成魔咒 解题报告
P4070 [SDOI2016]生成魔咒 题目描述 魔咒串由许多魔咒字符组成,魔咒字符可以用数字表示.例如可以将魔咒字符 \(1\).\(2\) 拼凑起来形成一个魔咒串 \([1,2]\). 一个魔咒 ...
- [Sdoi2016]生成魔咒[SAM or SA]
4516: [Sdoi2016]生成魔咒 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1017 Solved: 569[Submit][Statu ...
随机推荐
- katalon系列十:Katalon Studio自定义关键字之拖拽
Katalon Studio自带关键字“Drag And Drop To Object”,可以在这个网站实践:http://jqueryui.com/droppable/#default 不过“Dra ...
- axios封装(二)队列管理
在某些特定的场景(比如 即时搜索 ,表格分页),会频繁的发起ajax请求,而由于ajax是异步API,所以返回的时序并不能够保证,这时候就需要实现一个ajax队列,在相同的请求发起时,取消处理上一个请 ...
- [转]git学习------>git-rev-parse命令初识
git学习------>git-rev-parse命令初识 2017年06月13日 10:04:13 阅读数:2172 一.准备工作 第一步:在d盘git test目录下,新建工作区根目录dem ...
- Query类型_JDBC的方法_JAVA方法_Loadrunner脚本
数据库查询压力测试脚本 jdbc_java_查询类型接口测试 package com.test; import java.sql.Connection; import java.sql.DriverM ...
- rhel6 mysql skip-grant-tables 添加用户报错 ERROR 1290
不小心把数据库密码忘掉了, 这个时候我们只需要在数据库的配置文件里面添加 skip-grant-tables 然后重新启动服务,再登录数据库就不要我们输入密码了 这个时候我成功登录数据,可是不小心又把 ...
- Python学习小目录汇总
python其他知识目录 python基础知识-1 1.typora软件使用 2.python解释器安装 3.Python解释器环境变量添加 4.计算机编码知识: 5.输出print(): 6.变量 ...
- 5 种使用 Python 代码轻松实现数据可视化的方法
数据可视化是数据科学家工作中的重要组成部分.在项目的早期阶段,你通常会进行探索性数据分析(Exploratory Data Analysis,EDA)以获取对数据的一些理解.创建可视化方法确实有助于使 ...
- Android开发第二阶段(7)
今天:对项目的最后总结,宣传给下届学生做准备.为了更好的了解和深入书写本次项目的总结随笔.
- View 渲染
在Spring MVC 中,controllers不负责具体的页面渲染,仅仅是调用业务逻辑并返回model数据给view层,至于view层具体怎么展现,由专门的view层具体负责,这就是MVC模式,业 ...
- 【BioCode】将多个蛋白质序列分成单个的txt文档
代码说明: fasta格式的蛋白质序列,一个txt里面有很多蛋白质序列,计算ss.pssm或disorder score时候都需要单条计算,需要分开. 分割前: 分割后: show you the c ...