连NOI Day1T1都不会做。。。看了题解都写不出来还要抄Claris的代码。。

题目链接: (luogu)https://www.luogu.org/problemnew/show/P1117

(bzoj)https://www.lydsy.com/JudgeOnline/problem.php?id=4650

(uoj)http://uoj.ac/problem/219

题解:

\(f[i]\)表示以\(i\)结束的\(AA\)型子串个数,\(g[i]\)表示以\(i\)开始的\(AA\)型子串个数

怎么求\(f,g\)?

打破思维定势,谁说必须要一个一个求呢

分长度来求

枚举长度\(L\), 处理所有长度为\(2L\)的\(AA\)型子串对\(f\)和\(g\)的贡献

如果每隔\(L\)的长度放一个打点计时器,呸,关键点

那么任何\(AA\)型子串都会经过两个相邻关键点

首先肯定要满足这两个关键位置上的字符一样

在这个基础上求出往前往后最多多少个一样的

这个就转化成了LCP和LCS问题,并且两个相邻关键点对\(f,g\)数组的影响是区间+1,使用差分前缀和解决

然后推一推就行了,注意+1-1不要推错

时间复杂度为调和级数,\(O(n\log n)\)

代码

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#define llong long long
using namespace std; const int N = 1<<15;
const int LGN = 15;
const int S = 26;
int log2[N+3];
struct SparseTable
{
int n;
int str[N+3];
int rk[N+3];
int tmp[N+3];
int height[N+3];
int h[N+3];
int sa[N+3];
int wb[N+3];
int mini[N+3][LGN+3];
void get_sa()
{
int *x = rk,*y = tmp;
for(int i=0; i<=S; i++) wb[i] = 0;
for(int i=1; i<=n; i++) wb[x[i]=str[i]]++;
for(int i=1; i<=S; i++) wb[i] += wb[i-1];
for(int i=n; i>=1; i--) sa[wb[x[i]]--] = i;
int s = S,p = 0;
for(int j=1; p<n; j<<=1)
{
p = 0;
for(int i=n-j+1; i<=n; i++) y[++p] = i;
for(int i=1; i<=n; i++) if(sa[i]>j) y[++p] = sa[i]-j;
for(int i=1; i<=s; i++) wb[i] = 0;
for(int i=1; i<=n; i++) wb[x[y[i]]]++;
for(int i=1; i<=s; i++) wb[i] += wb[i-1];
for(int i=n; i>=1; i--) sa[wb[x[y[i]]]--] = y[i];
swap(x,y);
p = 1; x[sa[1]] = 1;
for(int i=2; i<=n; i++) x[sa[i]] = (y[sa[i]]==y[sa[i-1]] && y[sa[i]+j]==y[sa[i-1]+j]) ? p : ++p;
s = p;
}
for(int i=1; i<=n; i++) rk[sa[i]] = i;
for(int i=1; i<=n; i++)
{
h[i] = h[i-1]==0 ? 0 : h[i-1]-1;
while(i+h[i]<=n && sa[rk[i-1]]+h[i]<=n && str[i+h[i]]==str[sa[rk[i]-1]+h[i]])
{
h[i]++;
}
}
for(int i=1; i<=n; i++) height[i] = h[sa[i]];
for(int i=1; i<=n; i++) mini[i][0] = height[i];
for(int j=1; j<=LGN; j++)
{
for(int i=1; i+(1<<j)-1<=n; i++)
{
mini[i][j] = min(mini[i][j-1],mini[i+(1<<j-1)][j-1]);
}
}
}
int querymin(int lb,int rb)
{
int g = log2[rb-lb+1];
return min(mini[lb][g],mini[rb-(1<<g)+1][g]);
}
int LCP(int x,int y)
{
if(x==y) return n-x+1;
if(rk[x]>rk[y]) swap(x,y);
return querymin(rk[x]+1,rk[y]);
}
void clear()
{
for(int i=1; i<=n; i++) str[i] = rk[i] = tmp[i] = height[i] = h[i] = sa[i] = wb[i] = 0;
for(int i=1; i<=n; i++)
{
for(int j=0; j<=LGN; j++)
{
mini[i][j] = 0;
}
}
}
} s1,s2;
llong f[N+3];
llong g[N+3];
char a[N+3];
int n; int LCP(int x,int y) {return s1.LCP(x,y);}
int LCS(int x,int y) {return s2.LCP(n+1-x,n+1-y);} void preprocess()
{
log2[1] = 0; for(int i=2; i<=N; i++) log2[i] = log2[i>>1]+1;
} void clear()
{
s1.clear(); s2.clear();
for(int i=0; i<=n+1; i++) a[i] = 0,f[i] = g[i] = 0ll;
} int main()
{
preprocess();
int T; scanf("%d",&T);
while(T--)
{
scanf("%s",a+1); n = strlen(a+1); for(int i=1; i<=n; i++) a[i]-=96;
for(int i=1; i<=n; i++) s1.str[i] = a[i]; s1.n = n;
s1.get_sa();
for(int i=1; i<=n; i++) s2.str[i] = a[n+1-i]; s2.n = n;
s2.get_sa();
for(int i=1; i+i<=n; i++)
{
for(int j=i+i; j<=n; j+=i)
{
if(a[j]==a[j-i])
{
int lb = j-LCS(j,j-i)+1,rb = j+LCP(j,j-i)-1;
lb = max(lb+i-1,j); rb = min(rb,j+i-1);
if(lb<=rb)
{
f[lb]++; f[rb+1]--;
g[lb-i-i+1]++; g[rb+1-i-i+1]--;
}
}
}
}
for(int i=1; i<=n; i++) f[i] += f[i-1],g[i] += g[i-1];
llong ans = 0ll;
for(int i=1; i<n; i++)
{
llong tmp = f[i]*g[i+1];
ans += tmp;
}
printf("%lld\n",ans);
clear();
}
return 0;
}

UOJ #219 BZOJ 4650 luogu P1117 [NOI2016]优秀的拆分 (后缀数组、ST表)的更多相关文章

  1. UOJ#219. 【NOI2016】优秀的拆分 [后缀数组 ST表]

    #219. [NOI2016]优秀的拆分 题意:求有多少AABB样子的子串,拆分不同的同一个子串算多个 一开始一直想直接求,并不方便 然后看了一眼Claris的题解的第一行就有思路了 如果分开,求\( ...

  2. UOJ#219/BZOJ4650 [NOI2016]优秀的拆分 字符串 SA ST表

    原文链接http://www.cnblogs.com/zhouzhendong/p/9025092.html 题目传送门 - UOJ#219 (推荐,题面清晰) 题目传送门 - BZOJ4650 题意 ...

  3. [NOI2016]优秀的拆分 后缀数组

    题面:洛谷 题解: 因为对于原串的每个长度不一定等于len的拆分而言,如果合法,它将只会被对应的子串统计贡献. 所以子串这个限制相当于是没有的. 所以我们只需要对于每个位置i求出f[i]表示以i为开头 ...

  4. BZOJ.4650.[NOI2016]优秀的拆分(后缀数组 思路)

    BZOJ 洛谷 令\(st[i]\)表示以\(i\)为开头有多少个\(AA\)这样的子串,\(ed[i]\)表示以\(i\)结尾有多少个\(AA\)这样的子串.那么\(Ans=\sum_{i=1}^{ ...

  5. Luogu P1117 [NOI2016]优秀的拆分

    题目链接 \(Click\) \(Here\) 这题质量不错,就是暴力分有点足\(hhhhhhhh\),整整有\(95\)分. (搞得我写完暴力都不想写正解直接理解思路之后就直接水过去了\(QwQ\) ...

  6. BZOJ 4650 [Noi2016]优秀的拆分 ——后缀数组

    我们只需要统计在某一个点开始的形如$AA$字符串个数,和结束的个数相乘求和. 首先枚举循环节的长度L.即$\mid (A) \mid=L$ 然后肯定会经过s[i]和[i+L]至少两个点. 然后我们可以 ...

  7. BZOJ 4453: cys就是要拿英魂![后缀数组 ST表 单调栈类似物]

    4453: cys就是要拿英魂! Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 90  Solved: 46[Submit][Status][Discu ...

  8. bzoj 3796: Mushroom追妹纸【二分+后缀数组+st表】

    把三个串加上ASCII大于z的分隔符连起来,然后求SA 显然每个相同子串都是一个后缀的前缀,所以枚举s1的每个后缀的最长和s2相同的前缀串(直接在排序后的数组里挨个找,最近的两个分别属于s1和s2的后 ...

  9. [LOJ 2083][UOJ 219][BZOJ 4650][NOI 2016]优秀的拆分

    [LOJ 2083][UOJ 219][BZOJ 4650][NOI 2016]优秀的拆分 题意 给定一个字符串 \(S\), 求有多少种将 \(S\) 的子串拆分为形如 AABB 的拆分方案 \(| ...

随机推荐

  1. Git使用SSH提交代码到server出现 permission denied (publickey).

    在GitBush中向已经存在的Repository提交README.md改动. 命令例如以下: touch README.md git init git add README.md git commi ...

  2. 【BZOJ 1601】 灌水

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=1601 [算法] 最小生成树 [代码] #include<bits/stdc++ ...

  3. bzoj2303

    并查集+数学 这道题网上好像有两种解法. 这位写的很可读:http://blog.csdn.net/unicornt_/article/details/51901225 然后看完大概就懂了做法,但是实 ...

  4. Largest Rectangle in a Histogram(dp)

    http://acm.hdu.edu.cn/showproblem.php?pid=1506 题意:给出n个矩形的高度,每个矩形的宽都为1,求相邻的矩形能组合成的最大的矩形的面积. 思路:求出比第i个 ...

  5. 洛谷P1402 酒店之王(二分图)

    P1402 酒店之王 题目描述 XX酒店的老板想成为酒店之王,本着这种希望,第一步要将酒店变得人性化.由于很多来住店的旅客有自己喜好的房间色调.阳光等,也有自己所爱的菜,但是该酒店只有p间房间,一天只 ...

  6. 没调出来 P2023

    #include<iostream> #include<cstdio> #include<cstring> #define ll long long #define ...

  7. sqlserver导入数据到mysql的详细图解

    SQL Server 迁移数据到MySQL 一.背景 由于项目开始时候使用的数据库是SQL Server,后来把存储的数据库调整为MySQL,所以需要把SQL Server的数据转移到MySQL:由于 ...

  8. D - Garden

    Problem description Luba thinks about watering her garden. The garden can be represented as a segmen ...

  9. java中强制类型转换时,高位数截取成低位数的方法

    /** * 强制类型转换中的补码.反码.原码一搞清楚 */ int b=233;//正整数强转 System.out.println((byte)b); //负数:原码的绝对值取反再加一 符号为不变 ...

  10. 前端面试基础-html篇之H5新特性

    h5的新特性(目前个人所了解)如下 语义化标签 表单新特性 视频(video)和音频(audio) canvas画布 svg绘图 地理定位 为鼠标提供的拖放API webworker (重点)Stor ...