BZOJ4650: [Noi2016]优秀的拆分

时间:2022-12-22 12:14:54

考场上没秒的话多拿5分并不划算的样子。

思想其实很简单嘛。

要统计答案,求以每个位置开始和结束的AA串数量就好了。那么枚举AA中A的长度L,每L个字符设一个关键点,这样AA一定经过相邻的两个关键点。计算出相邻关键点的最长公共前后缀,把对应的位置区间加一下。

求lcp和lcs可以用后缀数组,也可以用hash。

UPD:UOJ上有人sxbk把原来的hash卡了,稍微改了下。其实base随机就没事了,然而BZOJ上并不能调用time(0)。

#include<bits/stdc++.h>
using namespace std;
typedef unsigned long long ll;
const int N=30005;
const int p=1e9+93;
ll f[N][2],g[N][2];
bool jud(int i1,int j1,int i2,int j2){
return f[j1][1]-f[i1][1]*g[j1-i1][1]==f[j2][1]-f[i2][1]*g[j2-i2][1]&&0==(f[j1][0]+(p-f[i1][0])*g[j1-i1][0]+p-f[j2][0]+f[i2][0]*g[j2-i2][0])%p;
}
char z[N];
int a[N],b[N];
int main(){
g[0][0]=1;
g[0][1]=1;
for(int i=1;i<N;++i){
g[i][0]=2003*g[i-1][0]%p;
g[i][1]=2011*g[i-1][1];
}
int q;
scanf("%d",&q);
while(q--){
scanf("%s",z+1);
int n=strlen(z+1);
for(int i=1;i<=n;++i){
f[i][0]=(z[i]+2003*f[i-1][0])%p;
f[i][1]=z[i]+2011*f[i-1][1];
}
fill_n(a+1,n,0);
fill_n(b+1,n,0);
for(int i=1;i<=n/2;++i)
for(int j=1;j<=n-i;j+=i)
if(z[j]==z[j+i]){
int l=0,r=min(i,j);
while(l!=r){
int m=l+r+1>>1;
if(jud(j-m,j,j+i-m,j+i))
l=m;
else
r=m-1;
}
int u=j-l+1;
l=0,r=min(i-1,n-j-i);
while(l!=r){
int m=l+r+1>>1;
if(jud(j,j+m,j+i,j+i+m))
l=m;
else
r=m-1;
}
int v=j+l+1;
if(u+i<=v){
++a[u];
--a[v-i+1];
++b[u+i*2-1];
--b[v+i];
}
}
ll s=0;
for(int i=1;i<=n;++i){
a[i]+=a[i-1];
b[i]+=b[i-1];
s+=a[i]*b[i-1];
}
printf("%lld\n",s);
}
}

BZOJ4650: [Noi2016]优秀的拆分的更多相关文章

  1. &lbrack;UOJ&num;219&rsqb;&lbrack;BZOJ4650&rsqb;&lbrack;Noi2016&rsqb;优秀的拆分

    [UOJ#219][BZOJ4650][Noi2016]优秀的拆分 试题描述 如果一个字符串可以被拆分为 AABBAABB 的形式,其中 A 和 B 是任意非空字符串,则我们称该字符串的这种拆分是优秀 ...

  2. BZOJ4650 &lbrack;NOI2016&rsqb;优秀的拆分 【后缀数组】

    题目 如果一个字符串可以被拆分为 AABBAABB 的形式,其中 AA 和 BB 是任意非空字符串,则我们称该字符串的这种拆 分是优秀的.例如,对于字符串 aabaabaa,如果令 A=aabA=aa ...

  3. UOJ&num;219&sol;BZOJ4650 &lbrack;NOI2016&rsqb;优秀的拆分 字符串 SA ST表

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

  4. BZOJ4650 NOI2016优秀的拆分(后缀数组)

    显然只要求出以每个位置开始的AA串数量就可以了,将其和反串同位置的结果乘一下,加起来就是答案.考虑对每种长度的字符串计数.若当前考虑的A串长度为x,我们每隔x个字符设一个关键点,求出相邻两关键点的后缀 ...

  5. &lbrack;BZOJ4650&rsqb;&lbrack;NOI2016&rsqb;优秀的拆分&lpar;SAM构建SA&rpar;

    关于解法这个讲的很清楚了,主要用了设关键点的巧妙思想. 主要想说的是一个刚学的方法:通过后缀自动机建立后缀树,再转成后缀数组. 后缀数组功能强大,但是最令人头疼的地方是模板太难背容易写错.用这个方法, ...

  6. bzoj千题计划317:bzoj4650&colon; &lbrack;Noi2016&rsqb;优秀的拆分(后缀数组&plus;差分)

    https://www.lydsy.com/JudgeOnline/problem.php?id=4650 如果能够预处理出 suf[i] 以i结尾的形式为AA的子串个数 pre[i] 以i开头的形式 ...

  7. 题解【bzoj4650 &lbrack;NOI2016&rsqb;优秀的拆分】

    Description 求对每一个连续字串将它切割成形如 AABB 的形式的方案数之和 Solution 显然 AABB 是由两个 AA 串拼起来的 考虑维护两个数组 a[i] 和 b[i] ,其中 ...

  8. BZOJ4650&colon; &lbrack;Noi2016&rsqb;优秀的拆分&lpar;hash 调和级数&rpar;

    题意 题目链接 Sol NOI的题都这么良心么.. 先交个\(n^4\)暴力 => 75 hash优化一下 => 90 然后\(90\)到\(100\)分之间至少差了\(10\)难度台阶= ...

  9. bzoj4650&colon; &lbrack;Noi2016&rsqb;优秀的拆分 hash

    好气啊,没开longlong又biubiu了 底层: 用hash或者奇奇怪怪的算法兹磁logn求最长公共前后缀 思路: 统计出从一个点开始和结束的形如AA的子串的个数 统计的时候把相邻的结果相乘加起来 ...

随机推荐

  1. centos6 安装mysql报错Requires&colon; libc&period;so&period;6&lpar;GLIBC&lowbar;2&period;14&rpar;

    是应为版本弄混了,不可以把el7的mysql装到el6系统上,重新下载centos6对应的版本的,这里是centos6选择el6版本的 wget http://dev.mysql.com/get/my ...

  2. WPF中获取鼠标相对于桌面位置

    var transform = PresentationSource.FromVisual(this).CompositionTarget.TransformFromDevice; var mouse ...

  3. div在不固定高度的情况下垂直或者水平居中

    方法一: 用一个"ghost"伪元素(看不见的伪元素)和 inline-block / vertical-align 可以搞定居中,非常巧妙.但是这个方法要求待居中的元素是 inl ...

  4. Phabricator部署手册

    参考:https://secure.phabricator.com/book/phabricator/article/installation_guide/ 概述 phabricator,由faceb ...

  5. hdu 4277 2012长春赛区网络赛 dfs&plus;hashmap &ast;&ast;&ast;

    hashmap判重大法好 #include<cstdio> #include<iostream> #include<algorithm> #include<c ...

  6. python中getattr函数 hasattr函数

    hasattr(object, name)作用:判断对象object是否包含名为name的特性(hasattr是通过调用getattr(ojbect, name)是否抛出异常来实现的).示例: &gt ...

  7. MongoDB之数据分布式存储

    在MongoDB的世界,做数据分布式存储显得非常简单.只要按照前面介绍的 集群搭建 完成就完全具备了数据分布式存储的要求. 在这里分清几个概念:去前面的文章可以找到介绍 1. 复制集   功能是实现数 ...

  8. proxool详细配置

    proxool详细配置 博客分类: Java 配置管理SQLServletprototypeXML  proxool一个数据库连接池框架,提供了对你选择的其它类型的驱动程序的连接池封装.可以非常简单的 ...

  9. C&plus;&plus;&sol;C&num;中堆栈、对象内存模型、深浅拷贝、Array&period;Clone方法

    转载自:http://blog.csdn.net/jarvischu/article/details/6425534 目录 1.      C++/C#中对象内存模型................. ...

  10. 对比两个同类型的泛型集合并返回差异泛型集合 ——两个List&lt&semi;类名&gt&semi;的比较

    1: /// <summary> 2: /// 对比两个同类型的泛型集合并返回差异泛型集合 3: /// </summary> 4: /// <typeparam nam ...