【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的常规方法是先抓取元素并保存到仓库,在脚本中需要用到的时候调取,但假如元素属性和个数是可变的,就不能事先保存到仓库了,需要在脚本运行时实时创建. 代码运行时实时创建一个元素对象的例子im ...
- 学习笔记之glog的使用
下载源码,使用cmake编译,最后得到了32位的静态库 glog.lib 使用库时要注意添加以下预定义: GLOG_NO_ABBREVIATED_SEVERITIES; GOOGLE_GLOG_DLL ...
- jquery中国地图插件
插件下载地址: http://www.17sucai.com/preview/1266961/2018-09-18/map/js/jsMap-1.1.0.min.js jsMap 项目介绍 这是一个功 ...
- 240. 搜索二维矩阵 II
二维数组搜索 编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target.该矩阵具有以下特性: 每行的元素从左到右升序排列. 每列的元素从上到下升序排列. 示例: 现有矩阵 ...
- 关于购买Redis服务器:腾讯云、阿里云还是华为云?
个人分类: redis使用 编辑 新年伊始,很多商家都开始进行新年产品大促销,在分布是缓存Redis领域,几家大公司也是打得如火如荼,各有千秋啊. 现在市场上比较有口碑的商家有腾讯云.阿里云.华为云三 ...
- 【python 2.7】python读取json数据存入MySQL
同上一篇,只是适配 CentOS+ python 2.7 #python 2.7 # -*- coding:utf-8 -*- __author__ = 'BH8ANK' import json im ...
- Liunx expect 基础
a script for study except #!/usr/bin/expect 声明文件内的语法使用 expect 的语法来执行. send send: 向进程发送字符串,用于模拟用户的输入. ...
- 网络流小结(HNOI2019之前)
\(\text{一:Dinic最大流}\) 最坏复杂度 \({\mathcal O(n^2m)}\) 一般可以处理 \(10^4\) ~ \(10^5\) 的网络. struct Edge { int ...
- Altera FPGA AS,PS,Jtag配置模式区别
Altera FPGA AS,PS,Jtag配置模式区别 FPGA器件有三类配置下载方式:主动配置方式(AS)和被动配置方式(PS)和最常用的(JTAG)配置方式. AS模式(active seri ...
- node.js常用方法
1.获取真实地址 function getClientIp(req) { return req.headers['x-forwarded-for'] || req.connection.remoteA ...