hihoCoder1449—后缀自动机三·重复旋律6

题目链接

时间限制: 15000ms
单点时限: 3000ms
内存限制: 512MB
描述

小Hi平时的一大兴趣爱好就是演奏钢琴。我们知道一个音乐旋律被表示为一段数构成的数列。

现在小Hi想知道一部作品中所有长度为K的旋律中出现次数最多的旋律的出现次数。但是K不是固定的,小Hi想知道对于所有的K的答案。

解题方法提示

输入

共一行,包含一个由小写字母构成的字符串S。字符串长度不超过 1000000。

输出

共Length(S)行,每行一个整数,表示答案。

样例输入
aab
样例输出
2
1
1


解题思路:

对S建立一个后缀自动机,我们可以先求出每个状态在原串中出现的次数cnt。对于复制的结点令cnt=0,转移的结点cnt=1,因为转移结点的状态保证至少有一个字符串,然后我们应该将一个状态的cnt值添加到它后缀链接的状态上,因为明显转移的状态的字符串被包含在原状态字符串中,所以我们可以用拓扑排序来更新状态的cnt值。

我们最后要求的是不同长度的字符串中出现次数的最大值,假设为ans[i],i从len(s)到1,ans肯定是非递减的,因为当前状态的字符串长于下一个状态的字符串,这意味着当前状态的字符串包含一个与下一个状态字符串等长的后缀。

所以有ans[i] = max(ans[i],ans[i+1]),ans的初始化为ans[st[i].len] = max(ans[st[i].len],st[i].cnt);

代码:

#include <iostream>
#include <algorithm>
#include <cstdio>
#include <cstring>
#include <string>
#include <map>
#include <queue>

using namespace std;

typedef long long llt;

const int N = 1000010;

struct state{
    int len,link,cnt;
    map<char,int>next;
}st[N*2];

int sz,last,ans[N],inDeg[N*2];

void sa_init()
{
    sz = last = 0;
    st[0].len = 0;
    st[0].link = -1;
    ++sz;

    for(int i = 0; i < N; ++i)
        st[i].next.clear(),st[i].cnt = 0;

    memset(ans,0,sizeof(ans));
    memset(inDeg,0,sizeof(inDeg));
}

void sa_extend(char c)
{
    int cur = sz++;         //新添加的状态结点
    st[cur].len = st[last].len+1;
    st[cur].cnt = 1;
    int p;
    for(p = last; p != -1 && !st[p].next.count(c); p = st[p].link)
        st[p].next[c] = cur;
    if(p == -1)
        st[cur].link = 0;
    else{
        int q = st[p].next[c];
        if(st[p].len+1 == st[q].len)
            st[cur].link = q,inDeg[q]++;
        else{
            int clone = sz++;
            st[clone].len = st[p].len+1;
            st[clone].next = st[q].next;
            st[clone].link = st[q].link;
            for(; p != -1 && st[p].next[c] == q; p = st[p].link)
                st[p].next[c] = clone;
            st[q].link = st[cur].link = clone;
            inDeg[clone] += 2;
        }
    }
    last = cur;
}

void bfs()
{
    queue<int>q;
    for(int i = 1; i < sz; ++i) if(!inDeg[i]) q.push(i);
    while(!q.empty()){
        int u = q.front(); q.pop();
        int v = st[u].link;
        ans[st[u].len] = max(ans[st[u].len],st[u].cnt);
        st[v].cnt += st[u].cnt;
        inDeg[v]--;
        if(!inDeg[v]) q.push(v);
    }
}

int main()
{
    sa_init();
    string str;
    cin >> str;
    for(int i = 0; i < str.length(); ++i)
        sa_extend(str[i]);

    bfs();

    for(int i = str.length(); i >= 1; --i) ans[i] = max(ans[i],ans[i+1]);

    for(int i = 1; i <= str.length(); ++i)
        cout << ans[i] << endl;
    return 0;
}

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值