Description

(我并不想告诉你题目名字是什么鬼)

有一个长度为n的仅包含小写字母的字符串S,下标范围为[1,n].

现在有若干组询问,对于每一个询问,我们给出若干个后缀(以其在S中出现的起始位置来表示),求这些后缀两两之间的LCP(LongestCommonPrefix)的长度之和.一对后缀之间的LCP长度仅统计一遍

Input

第一行两个正整数n,m,分别表示S的长度以及询问的次数.

接下来一行有一个字符串S.

接下来有m组询问,对于每一组询问,均按照以下格式在一行内给出:

首先是一个整数t,表示共有多少个后缀.接下来t个整数分别表示t个后缀在字符串S中的出现位置

Output

对于每一组询问,输出一行一个整数,表示该组询问的答案.由于答案可能很大,仅需要输出这个答案对于23333333333333333(一个巨大的质数)取模的余数.

Sample Input

7 3

popoqqq

1 4

2 3 5

4 1 2 5 6

Sample Output

0

0

2

### Hint
样例解释:
对于询问一,只有一个后缀”oqqq”,因此答案为0.
对于询问二,有两个后缀”poqqq”以及”qqq”,两个后缀之间的LCP为0,因此答案为0.
对于询问三,有四个后缀”popoqqq”,”opoqqq”,”qqq”,”qq”,其中只有”qqq”,”qq”两个后缀之间的LCP不为0,且长度为2,因此答案为2.
对于100%的测试数据,有\(S<=5*10^5\),且\(\sum t<=3*10^6\).
特别注意:由于另一世界线的某些参数发生了变化,对于一组询问,即使一个后缀出现了多次,也仅算一次.


首先一个很显然的思路就是直接把这个数组按照rank排序,然后我们发现对于每个点,前面的点的贡献从前往后是单调不减的,然后就可以直接维护单调栈了

挺水的题


#include<bits/stdc++.h>using namespace std;typedef pair<int, int> pi;
typedef long long ll;
const int N = 5e5 + 10;
const int M = 3e6 + 10;
const int LOG = 20;
const ll Mod = 23333333333333333;struct Suffix_Array {int s[N], n, m;int c[N], x[N], y[N];int height[N], sa[N], rank[N];int st[N][LOG], Log[N];ll sum[N]; void init(int len, char *c) {n = len, m = 0;for (int i = 1; i <= len; i++) {s[i] = c[i];m = max(m, s[i]);}}void radix_sort() {for (int i = 1; i <= m; i++) c[i] = 0;for (int i = 1; i <= n; i++) c[x[y[i]]]++;for (int i = 1; i <= m; i++) c[i] += c[i - 1];for (int i = n; i >= 1; i--) sa[c[x[y[i]]]--] = y[i];}void buildsa() {for (int i = 1; i <= n; i++) x[i] = s[i], y[i] = i;radix_sort();int now;for (int k = 1; k <= n; k <<= 1) {now = 0;for (int i = n - k + 1; i <= n; i++) y[++now] = i;for (int i = 1; i <= n; i++) if (sa[i] > k) y[++now] = sa[i] - k;radix_sort();y[sa[1]] = now = 1;for (int i = 2; i <= n; i++) y[sa[i]] = (x[sa[i]] == x[sa[i - 1]] && x[sa[i] + k] == x[sa[i - 1] + k]) ? now : ++now;swap(x, y);if (now == n) break;m = now;}}void buildrank() {for (int i = 1; i <= n; i++) rank[sa[i]] = i;}void buildsum() {for (int i = 1; i <= n; i++) sum[i] = sum[i - 1] + n - sa[i] + 1 - height[i];}void buildheight() {for (int i = 1; i <= n; i++) if (rank[i] != 1) {int k = max(height[rank[i - 1]] - 1, 0); // 里面是 rank[i - 1] for (; s[i + k] == s[sa[rank[i] - 1] + k]; k++);height[rank[i]] = k; // height 里面是 rank }}void buildst() {Log[1] = 0;for (int i = 2; i < N; i++) Log[i] = Log[i >> 1] + 1;for (int i = 1; i <= n; i++) st[i][0] = height[i];for (int j = 1; j < LOG; j++) {for (int i = 1; i + (1 << (j - 1)) <= n; i++) {st[i][j] = min(st[i][j - 1], st[i + (1 << (j - 1))][j - 1]);}}}int queryst(int l, int r) {if (l == r) return n - sa[l] + 1;if (l > r) swap(l, r);++l; //***int k = Log[r - l + 1];return min(st[l][k], st[r - (1 << k) + 1][k]);}int querylcp(int la, int lb) {return queryst(rank[la], rank[lb]);}void build(int len, char *c) {init(len, c);buildsa();buildrank();buildheight();buildsum();buildst();}
} Sa;int n, q, m;
char c[N];
ll ans, sum;
pi p[M];struct Node {int num, pos;ll val;
};
stack<Node> Q;ll add(ll a, ll b) {return (a += b) >= Mod ? a - Mod : a;
}int main() {
#ifdef dream_makerfreopen("input.txt", "r", stdin);
#endifscanf("%d %d", &n, &q);scanf("%s", c + 1);Sa.build(n, c); while (q--) {scanf("%d", &m);for (int i = 1; i <= m; i++) {scanf("%d", &p[i].second);p[i].first = Sa.rank[p[i].second];}sort(p + 1, p + m + 1);m = unique(p + 1, p + m + 1) - p - 1;ans = sum = 0;while (Q.size()) Q.pop();Q.push((Node) {1, p[1].second, 0});for (int i = 2; i <= m; i++) {int curnum = 1, len = Sa.querylcp(Q.top().pos, p[i].second);while (Q.size() && Q.top().val >= len) {curnum += Q.top().num;sum -= Q.top().val * Q.top().num;Q.pop();}Q.push((Node) {curnum, p[i].second, len});sum += len * curnum; ans = add(ans, sum);}printf("%lld\n", ans);}return 0;
}

转载于:https://www.cnblogs.com/dream-maker-yk/p/10072662.html

BZOJ3879: SvT【后缀数组+单调栈】相关推荐

  1. 【BZOJ3879】SvT,后缀数组+单调栈维护sum

    Time:2016.08.15 Author:xiaoyimi 转载注明出处谢谢 如果有不明白的地方可以在下面评论问我 传送门 思路: 建立后缀数组求出Height 如果说对每次询问暴力求LCP累加的 ...

  2. [BZOJ3238][AHOI2013]差异 [后缀数组+单调栈]

    题目地址 - GO-> 题目大意: 给定一个长度为 nn 的字符串SS,令TiTi表示它从第ii个字符开始的后缀,求以下这个式子的值: ∑1≤i<j≤nlen(Ti)+len(Tj)−2× ...

  3. [Ahoi2013]差异[后缀数组+单调栈]

    链接 解题思路:很明显前面∑1<=i<j<=nlen(Ti)+len(Tj)\sum_{1<=i<j<=n}len(T_i)+len(T_j)∑1<=i< ...

  4. [BZOJ 3238] [AHOI 2013] 差异 【后缀数组 + 单调栈】

    题目链接:BZOJ - 3238 题目分析 显然,这道题就是求任意两个后缀之间的LCP的和,这与后缀数组的联系十分明显. 求出后缀数组后,求出字典序相邻两个后缀的LCP,即 Height 数组. 那么 ...

  5. POJ - 3415 Common Substrings(后缀数组+单调栈)

    题目链接:点击查看 题目大意:给出两个字符串,再给出一个k,问两个字符串中长度大于等于k的公共子串有多少个(种类可重复) 题目分析:因为涉及到了子串问题,先用后缀数组跑出height数组来,接下来如果 ...

  6. [bzoj3238]差异(后缀数组+单调栈)

    显然我们可以先把len(Ti)+len(Tj)的值先算出来,再把LCP减去.所有len(Ti)+len(Tj)的值为n*(n-1)*(n+1)/2,这个随便在纸上画一画就可以算出来的. 接下来问题就是 ...

  7. POJ 3415 后缀数组+单调栈

    题目大意: 给定A,B两种字符串,问他们当中的长度大于k的公共子串的个数有多少个 这道题目本身理解不难,将两个字符串合并后求出它的后缀数组 然后利用后缀数组求解答案 这里一开始看题解说要用栈的思想,觉 ...

  8. bzoj 3238: [Ahoi2013]差异(后缀数组+单调栈)

    3238: [Ahoi2013]差异 Time Limit: 20 Sec  Memory Limit: 512 MB Submit: 3443  Solved: 1562 [Submit][Stat ...

  9. 【HAOI2016/BZOJ4566】找相同字符 后缀数组+单调栈

    原题走这里 鉴于我实在不是很懂单调栈和单调队列这一系列东西,所以我决定稍微具体讲一下单调栈. 恩,本题实质上就是求两个字符串的公共子串数,其中只要出现位置不同,就算是不同的子串. 处理多个字符串的经典 ...

最新文章

  1. Android Studio – Cannot resolve symbol ‘R’
  2. android fragment onHiddenChanged的使用
  3. debug ERP initial load error
  4. Ajax 编程基础(一)
  5. MySQL笔记-time类型的使用(hh:mm:ss)及time类型对比
  6. PHP创建圆柱体的类,创建一个类
  7. C#.NET高级面试题
  8. python3.7安装包下载_Python 3.7下载 Python 3.7 for win32 (Python编程开发工具)官方安装版 下载-脚本之家...
  9. 九章算术卷第二 粟米
  10. GridView 水平方向滑动
  11. 基于PHP图书馆图书借阅管理系统
  12. STC15单片机-LED闪烁(定时器)
  13. hdu 2502月之数
  14. 同时投掷两个不同面的骰子
  15. 枚举,随机数模型,函数功能分解---扔骰子随机数游戏
  16. js实现京东商城导航
  17. 【个人研究】21世纪西方运用脑电(EEG)的音乐研究动向分析(二)
  18. A. Greatest Convex
  19. Coursera吴恩达《优化深度神经网络》课程笔记(3)-- 超参数调试、Batch正则化和编程框架
  20. 逻辑函数的简化之图解法(卡诺图法)

热门文章

  1. python 数据分析
  2. 求1+2+……+N的和
  3. Ubuntu系统rm命令删除文件没有提示,怎么办?
  4. container_of 和 offsetof 宏详解
  5. 为什么Map桶中个数超过8才转为红黑树
  6. DSP学习 -- Visual Studio 操作
  7. Distributed Systems笔记-NFS、AFS、GFS
  8. Android设计模式MVVM之DataBinding简单使用
  9. JZOJ 5444. 【NOIP2017提高A组冲刺11.2】救赎
  10. JZOJ 3453【NOIP2013中秋节模拟】连通块