P4070 [SDOI2016]生成魔咒

题目地址:P4070 [SDOI2016]生成魔咒

相信看到题目之后很多人跟我的思路是一样的——

肯定要用 SAP3809 【模板】后缀排序

肯定要会求本质不同的子串个数P2408 不同子串个数

然后?就不会了......

瓶颈在哪儿?

你会发现每往后添加一个字符,整个 sa 数组只会插入一个数,要维护不难

但是 height无规律变化,这就导致无法高效维护

怎么办呢?

倒置字符串

我们将整个字符串倒置过来

显然本质不同的子串个数不会变化

而每往前添加一个字符串, height 的变化是 \(O(1)\) 的

那么,问题就变得简单很多了

具体实现请看代码注释

#include <bits/stdc++.h>
#define ll long long
#define si set<int>::iterator
using namespace std;
const int N = 1e5 + 6;
int n, m, a[N], b[N];
int sa[N], rk[N], tp[N], tx[N], he[N], st[N][20];
ll ans;
set<int> s;

inline void tsort() {//基数排序
    for (int i = 1; i <= m; i++) tx[i] = 0;
    for (int i = 1; i <= n; i++) ++tx[rk[i]];
    for (int i = 1; i <= m; i++) tx[i] += tx[i-1];
    for (int i = n; i; i--) sa[tx[rk[tp[i]]]--] = tp[i];
}

inline bool pd(int i, int w) {
    return tp[sa[i-1]] == tp[sa[i]] && tp[sa[i-1]+w] == tp[sa[i]+w];
}

inline void SA() {//后缀数组板子
    for (int i = 1; i <= n; i++) {
        rk[i] = a[i] = lower_bound(b + 1, b + m + 1, a[i]) - b;
        tp[i] = i;
    }
    tsort();
    for (int w = 1, p = 0; p < n; m = p, w <<= 1) {
        p = 0;
        for (int i = 1; i <= w; i++) tp[++p] = n - w + i;
        for (int i = 1; i <= n; i++)
            if (sa[i] > w) tp[++p] = sa[i] - w;
        tsort();
        swap(rk, tp);
        rk[sa[1]] = p = 1;
        for (int i = 2; i <= n; i++)
            rk[sa[i]] = pd(i, w) ? p : ++p;
    }
    int p = 0;
    for (int i = 1; i <= n; i++) {
        if (p) --p;
        int j = sa[rk[i]-1];
        while (a[i+p] == a[j+p]) ++p;
        he[rk[i]] = p;
    }
}

inline void ST() {//构造ST表
    for (int i = 1; i <= n; i++) st[i][0] = he[i];
    int w = log(n) / log(2);
    for (int k = 1; k <= w; k++)
        for (int i = 1; i <= n; i++) {
            if (i + (1 << k) > n + 1) break;
            st[i][k] = min(st[i][k-1], st[i+(1<<(k-1))][k-1]);
        }
}

inline int get(int l, int r) {//求l~r之间的最小值(即l-1与r的lcp)
    int k = log(r - l + 1) / log(2);
    return min(st[l][k], st[r-(1<<k)+1][k]);
}

int main() {
    cin >> n;
    for (int i = 1; i <= n; i++) {
        scanf("%d", &a[i]);
        b[i] = a[i];
    }
    //离散化
    sort(b + 1, b + n + 1);
    m = unique(b + 1, b + n + 1) - (b + 1);
    reverse(a + 1, a + n + 1);//倒置字符串
    SA();//求sa,rk,height数组
    ST();//ST表
    for (int i = n; i; i--) {//倒序考虑
        s.insert(rk[i]);//以rk为关键字插入set
        si it = s.find(rk[i]);//找到插入的位置
        int k = 0;//存最长lcp
        if (it != s.begin()) {//找前驱,注意特判
            int p = *(--it);
            k = get(p + 1, rk[i]);
            ++it;
        }
        ++it;
        if (it != s.end()) {//找后继,注意特判
            int p = *it;
            k = max(k, get(rk[i] + 1, p));
        }
        ans += n + 1 - i - k;//加上新生成的子串
        printf("%lld\n", ans);
    }
    return 0;
}

原文地址:https://www.cnblogs.com/xht37/p/10466980.html

时间: 2024-10-28 14:14:37

P4070 [SDOI2016]生成魔咒的相关文章

[Sdoi2016]生成魔咒[SAM or SA]

4516: [Sdoi2016]生成魔咒 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1017  Solved: 569[Submit][Status][Discuss] Description 魔咒串由许多魔咒字符组成,魔咒字符可以用数字表示.例如可以将魔咒字符 1.2 拼凑起来形成一个魔咒串 [1,2]. 一个魔咒串 S 的非空字串被称为魔咒串 S 的生成魔咒. 例如 S=[1,2,1] 时,它的生成魔咒有 [1].[2].[1,2].[2

BZOJ4516: [Sdoi2016]生成魔咒 后缀自动机

#include<iostream> #include<cstdio> #include<cstring> #include<queue> #include<cmath> #include<algorithm> #include<cstdlib> #include<map> #define N 200005 #define ll long long using namespace std; int read()

bzoj4516: [Sdoi2016]生成魔咒(SAM)

4516: [Sdoi2016]生成魔咒 题目:传送门 题解: 真奥义之SAM裸题... 其实对于当前新增节点x的操作,每次对ans的贡献就是dep[x]-dep[fail[x]](根据fail指针的定义随便YY) 然后有思路之后乍看题目每个x是10^9...瞬间GG %了已发cc然后被D飞,直接上map啊 代码: 1 #include<cstdio> 2 #include<cstring> 3 #include<cstdlib> 4 #include<cmath

[SDOI2016] 生成魔咒 - 后缀数组,平衡树,STL,时间倒流

[SDOI2016] 生成魔咒 Description 初态串为空,每次在末尾追加一个字符,动态维护本质不同的子串数. Solution 考虑时间倒流,并将串反转,则变为每次从开头删掉一个字符,即每次从后缀集合中删掉一个后缀. 预处理出后缀数组和高度数组后,用平衡树维护所有后缀集合(按照后缀排序),要删除一个后缀 \(S[sa[p],n]\) 时,找到它在平衡树上的前驱 \(u\) 和后继 \(v\) ,如果都存在,那么这一步的贡献就是 \[(n-sa[p]+1) - Max(h[p],h[v]

Bzoj4516 [Sdoi2016]生成魔咒

Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 947  Solved: 529 Description 魔咒串由许多魔咒字符组成,魔咒字符可以用数字表示.例如可以将魔咒字符 1.2 拼凑起来形成一个魔咒串 [1,2]. 一个魔咒串 S 的非空字串被称为魔咒串 S 的生成魔咒. 例如 S=[1,2,1] 时,它的生成魔咒有 [1].[2].[1,2].[2,1].[1,2,1] 五种.S=[1,1,1] 时,它的生成魔咒有 [1]. [1,1].[

BZOJ 4516: [Sdoi2016]生成魔咒

Description 给出一串数字,求每次插入一个数字后本质不同的子串. Sol SAM. 在 SAM 上添加节点的时候统计一下 \(val[np]-val[par[np]]\) 就可以了... 用 map 存一下边,复杂度 \(O(nlogn)\) Code /************************************************************** Problem: 4516 User: BeiYu Language: C++ Result: Accept

[SDOI2016]生成魔咒

OJ题号: BZOJ4516 题目大意: 按顺序在一个序列的末尾插入数字,每次求出插入后能得到的本质不同的子串个数. 思路: 每次在SAM后加入这个数字,每次新出现的本质不同的子串个数就等于new_p->len-new_p->link->len. 由于数字范围比较大,可以考虑离散化或者map. 事实上也可以用hash,不过实践证明会比map还慢很多,内存也浪费很多. 另外需要注意开long long. 1 #include<map> 2 #include<cstdio&

●BZOJ 4516 [Sdoi2016]生成魔咒

题链: http://www.lydsy.com/JudgeOnline/problem.php?id=4516 题解: 把串反过来后,问题变为求每个后缀的互不相同的子串个数.首先用倍增算法求出 sa[],rank[],height[],然后对 height[]数组建立 ST表.接着求出整个串的子串个数,ans+=N-sa[i]-height[i].(我从0开始编号的)式子的含义就是考虑每个后缀相比它的前一名,多了几个与之前不同的且串头为该后缀的头的子串. (一定要清晰地懂得并理解那个式子哦)

LibreOJ #2033. 「SDOI2016」生成魔咒

二次联通门 : LibreOJ #2033. 「SDOI2016」生成魔咒 /* LibreOJ #2033. 「SDOI2016」生成魔咒 调了整整一天啊... 绝望啊 最后发现是1打成i了啊!!! */ #include <iostream> #include <cstdio> #include <algorithm> #include <cmath> #include <map> const int BUF = 10000020; char