【AHOI2013】差异
题面
题解
$ \because \sum_{1 \leq i < j \leq n} i + j = \frac{n(n-1)(n+1)}2 $
所以只需求$\sum lcp(i,j)$即可。
$ \because lcp(i,j)=\min_{rank[i] \leq k \leq rank[j]}\{height[k]\} $
所以可以选用最小值分治算法:
int min[maxn];
long long ans;
void Div(int l, int r)
{
if(l == r) return (void)(ans += height[l]);
int mid = (l + r) >> 1;
Div(l, mid); Div(mid + 1, r);
min[mid] = height[mid];
min[mid + 1] = height[mid + 1];
for(RG int i = mid - 1; i >= l; i--)
min[i] = std::min(min[i + 1], height[i]);
for(RG int i = mid + 2; i <= r; i++)
min[i] = std::min(min[i - 1], height[i]);
int j = mid;
for(RG int i = mid; i >= l; i--)
{
while(j < r && min[j + 1] >= min[i]) ++j;
ans += 1ll * min[i] * (j - mid);
}
j = mid + 1;
for(RG int i = mid + 1; i <= r; i++)
{
while(j > l && min[j - 1] > min[i]) --j;
ans += 1ll * min[i] * (mid + 1 - j);
}
}
//...
Div(1, n);
但是我们要精益求精,我们可以想一想$O(n)$的算法。
用栈维护前面与$i$最近且小于等于$height[i]$的元素$p$
则转移方程为:
$ f[i]=f[p]+(i-p)\times height[i] $
//...
long long f[maxn];
struct node { int val, pos; };
std::stack<node> stk;
int main()
{
//...
long long ans = 0; int pos = 0;
for(RG int i = 1; i <= n; i++)
{
int p = pos;
while(!stk.empty() && stk.top().val > height[i]) stk.pop();
if(!stk.empty()) p = stk.top().pos;
ans += (f[i] = f[p] + (i - p) * height[i]);
if(!height[i]) pos = i;
stk.push((node){height[i], i});
}
}
代码
#include<cstdio>
#include<cstring>
#include<cctype>
#include<algorithm>
#include<stack>
#define RG register
#define file(x) freopen(#x".in", "r", stdin);freopen(#x".out", "w", stdout);
#define clear(x, y) memset(x, y, sizeof(x))
inline int read()
{
int data = 0, w = 1; char ch = getchar();
while(ch != '-' && (!isdigit(ch))) ch = getchar();
if(ch == '-') w = -1, ch = getchar();
while(isdigit(ch)) data = data * 10 + (ch ^ 48), ch = getchar();
return data * w;
}
const int maxn(500010);
char s[maxn];
int n, sa[maxn], rank[maxn], height[maxn];
void sort(int m)
{
static int t[maxn], t2[maxn], c[maxn];
int i, *x = t, *y = t2, p = 0;
std::fill(c + 1, c + m + 1, 0);
for(i = 1; i <= n; i++) ++c[x[i] = s[i]];
for(i = 1; i <= m; i++) c[i] += c[i - 1];
for(i = n; i; i--) sa[c[x[i]]--] = i;
for(RG int k = 1; k <= n && p < n; k <<= 1)
{
p = 0;
for(i = n - k + 1; i <= n; i++) y[++p] = i;
for(i = 1; i <= n; i++) if(sa[i] > k) y[++p] = sa[i] - k;
std::fill(c + 1, c + m + 1, 0);
for(i = 1; i <= n; i++) ++c[x[y[i]]];
for(i = 1; i <= m; i++) c[i] += c[i - 1];
for(i = n; i; i--) sa[c[x[y[i]]]--] = y[i];
std::swap(x, y), p = 1, x[sa[1]] = 1;
for(i = 2; i <= n; i++)
x[sa[i]] = (y[sa[i]] == y[sa[i - 1]]
&& y[sa[i] + k] == y[sa[i - 1] + k]) ? p : ++p;
m = p;
}
}
void get_height()
{
int k = 0;
for(RG int i = 1; i <= n; i++) rank[sa[i]] = i;
for(RG int i = 1; i <= n; i++)
{
if(k) --k;
int j = sa[rank[i] - 1];
while(s[i + k] == s[j + k]) ++k;
height[rank[i]] = k;
}
}
long long f[maxn];
struct node { int val, pos; };
std::stack<node> stk;
int main()
{
#ifndef ONLINE_JUDGE
file(cpp);
#endif
scanf("%s", s + 1); n = strlen(s + 1);
sort(130); get_height();
long long ans = 0; int pos = 0;
for(RG int i = 1; i <= n; i++)
{
int p = pos;
while(!stk.empty() && stk.top().val > height[i]) stk.pop();
if(!stk.empty()) p = stk.top().pos;
ans += (f[i] = f[p] + (i - p) * height[i]);
if(!height[i]) pos = i;
stk.push((node){height[i], i});
}
printf("%lld\n", 1ll * n * (n - 1) * (n + 1) / 2 - ans * 2);
return 0;
}
【AHOI2013】差异的更多相关文章
- BZOJ 3238: [Ahoi2013]差异 [后缀数组 单调栈]
3238: [Ahoi2013]差异 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2326 Solved: 1054[Submit][Status ...
- bzoj 3238 Ahoi2013 差异
3238: [Ahoi2013]差异 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2357 Solved: 1067[Submit][Status ...
- BZOJ 3238: [Ahoi2013]差异 [后缀自动机]
3238: [Ahoi2013]差异 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2512 Solved: 1140[Submit][Status ...
- BZOJ_3238_[Ahoi2013]差异_后缀自动机
BZOJ_3238_[Ahoi2013]差异_后缀自动机 Description Input 一行,一个字符串S Output 一行,一个整数,表示所求值 Sample Input cacao Sam ...
- BZOJ_3238_[Ahoi2013]差异_后缀数组+单调栈
BZOJ_3238_[Ahoi2013]差异_后缀数组+单调栈 Description Input 一行,一个字符串S Output 一行,一个整数,表示所求值 Sample Input cacao ...
- 【LG4248】[AHOI2013]差异
[LG4248][AHOI2013]差异 题面 洛谷 题解 后缀数组版做法戳我 我们将原串\(reverse\),根据后缀自动机的性质,两个后缀的\(lcp\)一定是我们在反串后两个前缀的\(lca\ ...
- 【BZOJ3238】[AHOI2013]差异
[BZOJ3238][AHOI2013]差异 题面 给定字符串\(S\),令\(T_i\)表示以它从第\(i\)个字符开始的后缀.求 \[ \sum_{1\leq i<j\leq n}len(T ...
- P4248 [AHOI2013]差异 解题报告
P4248 [AHOI2013]差异 题目描述 给定一个长度为 \(n\) 的字符串 \(S\),令 \(T_i\) 表示它从第 \(i\) 个字符开始的后缀.求 \[\displaystyle \s ...
- 【BZOJ 3238】 3238: [Ahoi2013]差异(SAM)
3238: [Ahoi2013]差异 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 3047 Solved: 1375 Description In ...
- bzoj 3238: [Ahoi2013]差异 -- 后缀数组
3238: [Ahoi2013]差异 Time Limit: 20 Sec Memory Limit: 512 MB Description Input 一行,一个字符串S Output 一行,一个 ...
随机推荐
- URAL-1019 Line Painting----暴力或线段树
题目链接: https://cn.vjudge.net/problem/URAL-1019 题目大意: 一个0~1e9的区间,初始都是白的,现进行N次操作,每次将一段区间图上一中颜色.最后问说连续最长 ...
- iOS绘图事务的运行验证
结合WWDC,以我们的call stack为例,来说明这四个过程分别大概都做了什么. layout过程 从上面layout的过程可以看出,其所做的主要任务就是将图层调用代理(也就是视图)实现整个视图层 ...
- Mac eclipse导入项目中文乱码问题解决
方法一 1.打开eclipse 偏好设置 2.General ——>Content Types——>Text——>Java SourceFile 3.将编码设置为GBK. 4.upd ...
- unbuntu 14安装 golang
golang目前有两种编译,一种是golang官方提供的,另外一个是gnu提供的gccgo.这里安装的是Golang,从仓库安装(apt-get) sudo apt-get instal ...
- 【[ZJOI2008]泡泡堂】
想贪心就是反复\(hack\)自己的过程 首先这很田忌赛马,但是又不完全一样 田忌赛马保证了所有马的实力不同,因此没有平局 田忌赛马的策略是当自己最强的马比不过对方最强的马的时候,就用自己最弱的马来自 ...
- VC++获取一个GB级大文件的字节大小
常规的获得小文件(2.1GB以下)的字节大小可以使用ftell,函数 ftell 用于得到文件位置指针当前位置相对于文件首的偏移字节数.使用fseek函数后再调用函数ftell()就能非常容易地确定文 ...
- 【转】上传jar包到nexus私服
原文:https://my.oschina.net/lujianing/blog/297128 1通过网页上传 这种方法只是上传了jar包.通过maven引用当前jar,不能取得jar的依赖 from ...
- 页面QQ在线咨询、在线交谈代码
页面QQ在线咨询.在线交谈代码 样式一: <a target="blank" rel="nofollow" href="tencent://me ...
- javascript中获取dom元素高度和宽度
javascript中获取dom元素高度和宽度的方法如下: 网页可见区域宽: document.body.clientWidth网页可见区域高: document.body.clientHeight网 ...
- loadrunner脚本中参数化和返回值输出log到外部文件
loadrunner脚本中参数化和返回值输出log到外部文件 很多时候,我们在做性能测试之前,需要造数据,但是使用的这些参数化数据和生成的返回数据在后面的测试都会用的,所以我们需要在造数据过程中,将参 ...