UOJ#219. 【NOI2016】优秀的拆分 [后缀数组 ST表]
#219. 【NOI2016】优秀的拆分
题意:求有多少AABB样子的子串,拆分不同的同一个子串算多个
一开始一直想直接求,并不方便
然后看了一眼Claris的题解的第一行就有思路了
如果分开,求\(f[i]\)以i结尾AA形式子串和\(g[i]\)以i开始AA形式子串 就可以套路了
使用常用技巧,枚举\(L=|A|\),AA子串一定覆盖了两个关键点,枚举更新就行了,对于区间加可以使用差分
其实这道题很好拿95分啊,\(O(n^2)\)用哈希判断就行了
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
const int N=3e4+5;
typedef long long ll;
inline int read(){
char c=getchar();int x=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
return x*f;
}
int n, Log[N]; char s[N];
namespace ST {
void build(int f[N][16], int *a) {
for(int i=1; i<=n; i++) f[i][0]=a[i];
for(int j=1; j<15; j++)
for(int i=1; i+(1<<j)-1<=n; i++)
f[i][j] = min(f[i][j-1], f[i+(1<<(j-1))][j-1]);
}
}
struct SA {
int sa[N], t1[N], t2[N], c[N], rnk[N], hei[N], f[N][16];
inline bool cmp(int *r, int a, int b, int j) {
return a+j<=n && b+j<=n && r[a]==r[b] && r[a+j]==r[b+j];
}
void build(char *s, int m) {
int *r=t1, *k=t2;
for(int i=0; i<=m; i++) c[i]=0;
for(int i=1; i<=n; i++) c[r[i]=s[i]]++;
for(int i=1; i<=m; i++) c[i] += c[i-1];
for(int i=n; i>=1; i--) sa[ c[r[i]]-- ]=i;
for(int j=1; j<=n; j<<=1) {
int p=0;
for(int i=n-j+1; i<=n; i++) k[++p]=i;
for(int i=1; i<=n; i++) if(sa[i]>j) k[++p]=sa[i]-j;
for(int i=0; i<=m; i++) c[i]=0;
for(int i=1; i<=n; i++) c[r[k[i]]]++;
for(int i=1; i<=m; i++) c[i] += c[i-1];
for(int i=n; i>=1; i--) sa[ c[r[k[i]]]-- ]=k[i];
swap(r, k); p=0; r[sa[1]]=++p;
for(int i=2; i<=n; i++) r[sa[i]] = cmp(k, sa[i], sa[i-1], j) ? p : ++p;
if(p>=n) break; m=p;
}
int now=0;
for(int i=1; i<=n; i++) rnk[sa[i]]=i;
for(int i=1; i<=n; i++) {
if(now) now--;
if(rnk[i]==1) continue;
int j=sa[rnk[i]-1];
while(i+now<=n && j+now<=n && s[i+now]==s[j+now]) now++;
hei[rnk[i]]=now;
}
ST::build(f, hei);
}
int lcp(int x, int y) {
x=rnk[x], y=rnk[y];
if(x>y) swap(x, y); x++;
int t=Log[y-x+1];
return min(f[x][t], f[y-(1<<t)+1][t]);
}
}a, b;
inline int lcp(int x, int y) {return a.lcp(x, y);}
inline int lcs(int x, int y) {return b.lcp(n-x+1, n-y+1);}
int f[N], g[N];
inline void add(int *d, int l, int r) {d[l]++; d[r+1]--;}
void solve(int L) { //printf("\nsolve %d\n",L);
for(int i=1; i+L<=n; i+=L) {
int l = i - lcs(i, i+L) + 1, r = i + L + lcp(i, i+L) - 1;
l = max(l, i-L+1); r = min(r, i+L+L-1);
l = max(l, 1); r = min(r, n);
//printf("key %d %d [%d, %d]\n", i, i+L, l, r);
if(r-l+1 < 2*L) continue;
add(f, l+2*L-1, r); add(g, l, r-2*L+1);
}
}
int main() {
freopen("in","r",stdin);
Log[1]=0; for(int i=2; i<N; i++) Log[i] = Log[i>>1]+1;
int T=read();
while(T--) {
scanf("%s", s+1); n=strlen(s+1);
a.build(s, 260); reverse(s+1, s+1+n); b.build(s, 260); reverse(s+1, s+1+n);
for(int i=1; i<=n; i++) f[i]=g[i]=0;
//for(int i=1; i<=n; i++) for(int j=i; j<=n; j++) printf("lcs %d %d %d\n",i,j,lcs(i,j));
for(int i=1; i<=n; i++) solve(i);
ll ans=0;
for(int i=1; i<=n; i++) f[i]+=f[i-1], g[i]+=g[i-1];// printf("look %d %d %d\n",i,f[i],g[i]);
for(int i=2; i<n; i++) ans += (ll)f[i]*g[i+1];
printf("%lld\n", ans);
}
}
UOJ#219. 【NOI2016】优秀的拆分 [后缀数组 ST表]的更多相关文章
- UOJ#219/BZOJ4650 [NOI2016]优秀的拆分 字符串 SA ST表
原文链接http://www.cnblogs.com/zhouzhendong/p/9025092.html 题目传送门 - UOJ#219 (推荐,题面清晰) 题目传送门 - BZOJ4650 题意 ...
- [NOI2016]优秀的拆分 后缀数组
题面:洛谷 题解: 因为对于原串的每个长度不一定等于len的拆分而言,如果合法,它将只会被对应的子串统计贡献. 所以子串这个限制相当于是没有的. 所以我们只需要对于每个位置i求出f[i]表示以i为开头 ...
- UOJ #219 BZOJ 4650 luogu P1117 [NOI2016]优秀的拆分 (后缀数组、ST表)
连NOI Day1T1都不会做...看了题解都写不出来还要抄Claris的代码.. 题目链接: (luogu)https://www.luogu.org/problemnew/show/P1117 ( ...
- BZOJ.4650.[NOI2016]优秀的拆分(后缀数组 思路)
BZOJ 洛谷 令\(st[i]\)表示以\(i\)为开头有多少个\(AA\)这样的子串,\(ed[i]\)表示以\(i\)结尾有多少个\(AA\)这样的子串.那么\(Ans=\sum_{i=1}^{ ...
- BZOJ 4650 [Noi2016]优秀的拆分 ——后缀数组
我们只需要统计在某一个点开始的形如$AA$字符串个数,和结束的个数相乘求和. 首先枚举循环节的长度L.即$\mid (A) \mid=L$ 然后肯定会经过s[i]和[i+L]至少两个点. 然后我们可以 ...
- SPOJ 687 Repeats(后缀数组+ST表)
[题目链接] http://www.spoj.com/problems/REPEATS/en/ [题目大意] 求重复次数最多的连续重复子串的长度. [题解] 考虑错位匹配,设重复部分长度为l,记s[i ...
- POJ 3693 Maximum repetition substring(后缀数组+ST表)
[题目链接] poj.org/problem?id=3693 [题目大意] 求一个串重复次数最多的连续重复子串并输出,要求字典序最小. [题解] 考虑错位匹配,设重复部分长度为l,记s[i]和s[i+ ...
- BZOJ_4516_[Sdoi2016]生成魔咒_后缀数组+ST表+splay
BZOJ_4516_[Sdoi2016]生成魔咒_后缀数组+ST表+splay Description 魔咒串由许多魔咒字符组成,魔咒字符可以用数字表示.例如可以将魔咒字符 1.2 拼凑起来形成一个魔 ...
- UVA10829 L-Gap Substrings(后缀数组+ST表)
后缀数组+ST表. 代填的坑. \(Code\ Below:\) #include <bits/stdc++.h> #define ll long long using namespace ...
随机推荐
- 在64位系统下,指向int型的指针占的内存空间多大?
不废话,请看代码演示如下: 注意使用的操作系统的位数,不同位数的操作系统,结果不一样! 我是用的是64位的操作系统! linux下示例代码如下: #include <stdio.h> in ...
- JAVA经典算法面试40题及答案
现在是3月份,也是每年开年企业公司招聘的高峰期,同时有许多的朋友也出来找工作.现在的招聘他们有时会给你出一套面试题或者智力测试题,也有的直接让你上机操作,写一段程序.算法的计算不乏出现,基于这个原因我 ...
- [国嵌攻略][179][OpenSSL加密系统]
未加密传输的安全弊端 如果在网络传输中没有加密,就是以明文传输.传输的数据可以被抓包软件直接截获,并能读取里面的数据. 加密基本原理 1.对称加密 2.非对称加密 2.1.公钥私钥 公钥和私密要配对. ...
- sql for xml 输出结果带单引号出现转成&apos的解决方案
select '''' + ID +''',' from 表 for xml path('') 此SQL语句,输出结果如‘1’,’2‘,’3‘, 但是在因xml会出现path转译的问题将‘转成&am ...
- SSL证书安装指引
https://cloud.tencent.com/document/product/400/4143 下载得到的 www.domain.com.zip 文件,解压获得3个文件夹,分别是Apache. ...
- 评论发布信息可插入QQ表情
demo例子: HTML文本内容: <template> <div id="publish"> <!-- 发布内容输入框,利用Html5的新属性con ...
- PostgreSQL9.6.2的WINDOWS下安装
下载链接:https://www.enterprisedb.com/downloads/postgres-postgresql-downloads#windows 安装链接:http://www.cn ...
- php 利用Gd库添加文字水印乱码的问题及解决方案
最近一个项目进行了服务器迁移,部署后发现 ,其中一个为图片添加水印文字的功能出现了乱码问题,确认功能代码不存在问题,同时项目代码都是使用UTF-8编码,不存在编码问题,也检查排除了字体文件出现问题的可 ...
- python3 第二章 - 第一个程序
1、安装 打开官网 https://www.python.org/downloads/ 下载python3.6.4 如果你是windows\mac电脑,直接双击安装包,一路next即可,如果你是lin ...
- scrapy_移除内容中html标签
如何移除所获取内容中多余的html标签? 通过w3lib模块和re模块 #!/usr/bin/python3 # -*- coding: UTF-8 -*- __author__ = 'beimenc ...