【链接】h在这里写链接


【题意】


    给你一个长度为n的数组a[]

    设b[i] = a[i+1]-a[i];

    然后让你在b[i]里面找ABA的形式。

    这里B的长度要求为m;

    然后让你找这样的连续段的个数。

    n<=10^5,1<=m<=10

【题解】


    枚举A的长度i;

    然后把长度为N的数组分成i份i份的。

    即1..i,i+1..2i,2i+1..3i..

        关键点->i,2i,3i

        枚举每一个关键点->x

            然后获取x+m+i设为y



            则有:

                x,x+1,...x+m,x+m+1..x+m+i

                x                     y

            然后,让x和y同时往左走,设长度为len1.

            让x和y同时往右走,设长度为len2.

            (走的时候要时刻保持s[x]==s[y])

            如果len1+len2>=i,则对答案的贡献为len1+len2-i+1;

            向左向右扩展都不要超过x。这样才不会重复计数。

            只要获取x和x+m+i的lcp,就能得到len2;

            把整个字符串倒过来,再求n-x+1和n-(x+m+i)+1的后缀,就能得到len1了.

            可以一开始就把倒着的和顺着的用一个连接符连成一个字符串.



            枚举A的长度,然后在过程里面做对应答案的就好。

            需要写一个离散化。

【错的次数】


0

【反思】


最长公共前缀,把字符串倒一下就是求后缀啦

【代码】

#include <bits/stdc++.h>
#define ll long long
using namespace std; const int N = 1e5 + 100;
const int MAX_CHAR = 5e4 + 100;//每个数字的最大值。
int s[N + 10],n,m;//如果是数字,就写成int s[N+10]就好,从0开始存
int Sa[N + 10], T1[N + 10], T2[N + 10], C[N+10];
int Height[N + 10], Rank[N + 10];
vector <int> v; void build_Sa(int n, int m) {
int i, *x = T1, *y = T2;
for (i = 0; i<m; i++) C[i] = 0;
for (i = 0; i<n; i++) C[x[i] = s[i]]++;
for (i = 1; i<m; i++) C[i] += C[i - 1];
for (i = n - 1; i >= 0; i--) Sa[--C[x[i]]] = i;
for (int k = 1; k <= n; k <<= 1)
{
int p = 0;
for (i = n - k; i<n; i++) y[p++] = i;
for (i = 0; i<n; i++) if (Sa[i] >= k) y[p++] = Sa[i] - k;
for (i = 0; i<m; i++) C[i] = 0;
for (i = 0; i<n; i++) C[x[y[i]]]++;
for (i = 1; i<m; i++) C[i] += C[i - 1];
for (i = n - 1; i >= 0; i--) Sa[--C[x[y[i]]]] = y[i];
swap(x, y);
p = 1; x[Sa[0]] = 0;
for (i = 1; i<n; i++)
x[Sa[i]] = y[Sa[i - 1]] == y[Sa[i]] && y[Sa[i - 1] + k] == y[Sa[i] + k] ? p - 1 : p++;
if (p >= n) break;
m = p;
}
} void getHeight(int n)
{
int i, j, k = 0;
for (i = 1; i <= n; i++) Rank[Sa[i]] = i;
for (i = 0; i<n; i++) {
if (k) k--;
j = Sa[Rank[i] - 1];
while (s[i + k] == s[j + k]) k++;
Height[Rank[i]] = k;
}
} const int MAXL = 18;//log2数组的最大长度
const int INF = 0x3f3f3f3f;//数值绝对值的最大值 struct abc{
int pre2[MAXL+5],need[N+10];
int fmax[N+10][MAXL+5],fmin[N+10][MAXL+5]; void init(int n)
{
pre2[0] = 1;
for (int i = 1;i <= MAXL;i++)
{
pre2[i] = pre2[i-1]<<1;
}
need[1] = 0; need[2] = 1;
int temp = 2;
for (int i = 3; i <= n; i++)//need[i]表示长度为i是2的多少次方,可以理解为[log2i]
if (pre2[temp] == i)
need[i] = need[i - 1] + 1, temp++;
else
need[i] = need[i - 1];
} void getst(int *a,int n)
{
memset(fmax,-INF,sizeof fmax);
memset(fmin,INF,sizeof fmin);
for (int i = 1;i <= n;i++)//下标从0开始就改成对应的就好
fmax[i][0] = fmin[i][0] = a[i]; for (int l = 1;pre2[l]<=n;l++)
for (int i = 1;i <= n;i++)
if (i+pre2[l]-1<=n)
fmax[i][l] = max(fmax[i][l-1],fmax[i+pre2[l-1]][l-1]); for (int l = 1;pre2[l]<=n;l++)
for (int i = 1;i <= n;i++)
if (i+pre2[l]-1<=n)
fmin[i][l] = min(fmin[i][l-1],fmin[i+pre2[l-1]][l-1]);
} int getmin(int l,int r)
{
int len = need[r-l+1];
return min(fmin[l][len],fmin[r-pre2[len]+1][len]);
} int getmax(int l,int r)
{
int len = need[r-l+1];
return max(fmax[l][len],fmax[r-pre2[len]+1][len]);
} }ST; int lcp(int x,int y)
{
return ST.getmin(x+1,y);
} ll ans; void solve(int L)
{
for (int x = L-1;x + m + L <n;x+=L)
{
int y = x + m + L;
int len1 = lcp(min(Rank[x],Rank[y]),max(Rank[x],Rank[y]));
//往右走的lcp
len1 = min(len1,L); int len2 = lcp(min(Rank[2*n-x],Rank[2*n-(x+m+L)]),max(Rank[2*n-x],Rank[2*n-(x+m+L)]));
len2 = min(len2,L);
//往左走的LCp int temp = len1+len2;
if (len1>0 && len2 > 0) temp--;
if (temp >= L)
{
ans += temp-L+1;
}
}
} int main()
{
//freopen("F:\\rush.txt","r",stdin);
scanf("%d%d",&n,&m);
for (int i = 0;i <= n-1;i++) scanf("%d",&s[i]);
n--;
for (int i = 0;i <= n-1;i++){
s[i] = s[i+1]-s[i];
v.push_back(s[i]);
}
sort(v.begin(),v.end());
for (int i = 0;i < n;i++)
s[i] = lower_bound(v.begin(),v.end(),s[i]) - v.begin() + 1;
s[n] = 5e4 + 1;
int temp = n-1;//temp指向输入的最后一个数字
//总的长度为n*2 + 1 最后一个数字的位置就是2*n
for (int i = n+1;temp>=0;i++) {
s[i] = s[temp--];
}
s[2*n+1] = 0; build_Sa(2*n + 1 + 1,MAX_CHAR);
getHeight(2*n+1);
ST.init(2*n+1);
ST.getst(Height,2*n+1); for (int i = 1;i-1+m+i < n;i++)//枚举A的长度
solve(i);
printf("%lld\n",ans);
return 0;
}

【BZOJ 2119】股市的预测的更多相关文章

  1. BZOJ 2119: 股市的预测 [后缀数组 ST表]

    2119: 股市的预测 Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 331  Solved: 153[Submit][Status][Discuss ...

  2. BZOJ 2119: 股市的预测 SA

    2119: 股市的预测 Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 434  Solved: 200[Submit][Status][Discuss ...

  3. bzoj 2119: 股市的预测

    Description 墨墨的妈妈热爱炒股,她要求墨墨为她编写一个软件,预测某只股票未来的走势.股票折线图是研究股票的必备工具,它通过一张时间与股票的价位的函数图像清晰地展示了股票的走势情况.经过长时 ...

  4. ●BZOJ 2119 股市的预测

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=2119 题解: 这个题很好的. 首先把序列转化为差分序列,问题转化为找到合法的子序列,使得去除 ...

  5. bzoj 2119 股市的预测——枚举长度的关键点+后缀数组

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2119 就是找差分序列上中间差 m 的相等的两段. 考虑枚举这样一段的长度 L .可以把序列分 ...

  6. bzoj 2119 股市的预测 —— 枚举关键点+后缀数组

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2119 思路就是对于这个形如 ABA 的串,枚举 A 的长度,并按照长度分出几块,找到一些关键 ...

  7. BZOJ 2119 股市的预测 (后缀数组+RMQ)

    题目大意:求一个字符串中形如$ABA$的串的数量,其中$B$的长度是给定的 有点像[NOI2016]优秀的拆分这道题 先对序列打差分,然后离散,再正反跑$SA$,跑出$st$表 进入正题 $ABA$串 ...

  8. BZOJ 2119 股市的预测(后缀数组)

    首先要差分+离散化. 然后就是求形如ABA的串有多少,其中B的长度确定为k. 我们用到了设置关键点的思想.我们枚举A的长度L.然后在\(1,1+L,1+L*2,1+L*3...\)设置关键点.然后我们 ...

  9. BZOJ 2119: 股市的预测 (Hash / 后缀数组 + st表)

    转博客大法好 自己画一画看一看,就会体会到这个设置关键点的强大之处了. CODE(sa) O(nlogn)→1436msO(nlogn)\to 1436msO(nlogn)→1436ms #inclu ...

  10. 【BZOJ 2119】 2119: 股市的预测 (后缀数组+分块+RMQ)

    2119: 股市的预测 Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 404  Solved: 188 Description 墨墨的妈妈热爱炒股,她 ...

随机推荐

  1. html元素的分类有哪些?

    今天零度给大家讲一下基本的html元素分类: HTML元素的分类其实主要有两种元素构成——块级元素和内联元素. html元素的分类有块级元素和行内元素 一.块级元素(block)的特点: 1.总是在新 ...

  2. 【基础篇】EditText的一些属性设置

    设置EditText的背景颜色  private test_editText=null; test_editText= (EditText) findViewById(R.id.EditTextInp ...

  3. Kinect 开发 —— Hello,Kinect

    控制台输出深度数据: using System; using System.Collections.Generic; using System.Linq; using System.Text; usi ...

  4. Codefroces Educational Round 27 845G Shortest Path Problem?

    Shortest Path Problem? You are given an undirected graph with weighted edges. The length of some pat ...

  5. batch---系统不繁忙时执行任务

    batch:不需要指定时间,自动在系统空闲的时候执行指定的任务 [root@xiaolizi ~]# batch at> echo 1234at> <EOT>job 5 at ...

  6. 洛谷 P2958 [USACO09OCT]木瓜的丛林Papaya Jungle

    P2958 [USACO09OCT]木瓜的丛林Papaya Jungle 题目描述 Bessie has wandered off the farm into the adjoining farmer ...

  7. git 工具的使用总结(6)-提交合并处理

    1.撤消修改 1)revert:反转提交,它就是把你的一个提交先撤消掉,但是,它跟reset不同的是,你的这次这小会留下记录,这样在你下次需要的时候,可以通过这个节点把撤消的提交恢复 zhangshu ...

  8. [NOI.AC#33]bst 线段树

    链接 区间修改,完全二叉树,这引导我们把这棵树看成一棵线段树 .线段树的每一个节点相当于这棵二叉树的节点, 对于区间交换操作,我们对二叉树的每一层从上到下分别考虑,找到L,R在第i层对应的节点修改 这 ...

  9. 2.Java实现基于SOAP的XML文档网络传输及远程过程调用(RPC)-

    转自:https://blog.csdn.net/a214919447/article/details/55260411 SOAP(Simple Object Access Protocol,简单对象 ...

  10. BZOJ1195: [HNOI2006]最短母串(Trie图,搜索)

    Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. Input 第一行是一个正整数n(n<=12) ...