codeforces 645 E. Intellectual Inquiry

一个字符串,由前k个字母组成,长度为m + n,其中前m个字符已经确定,后面n个由你自由选择,

使得这个串的不同的子序列的个数最多,空串也算一个子序列。

1 <= m <= 10^6,0 <= n <= 10^6,1 <= k <= 26

首先,我们考虑n = 0的情况,

问题就为给定一个字符串,求它有多少个不同的子序列。

pre[i]表示字母i最后出现的位置,初始化为0

f[i]表示以第i个字符结尾的与前面已经出现的子序列都不同的子序列个数

g[i] = ∑0<=j<=if[j]

则我们知道f[i] = ∑pre[i]<=j<=i-1f[j] 

则 if pre[i] == 0  then f[i] = g[i-1]

    if pre[i] > 0    then f[i] = g[i-1] - g[pre[str[i]]-1]

答案就是g[m]

那如果n > 0 呢?

从f的递推式我们知道,要使得f[i]最大,i处填的字符应该是pre值最小的i

那么我们接着遍历,每次拿pre最小的字符,更新f和pre值

最终答案就是g[n + m]

 

代码:

                                           
  //File Name: cf645E.cpp
  //Created Time: 2017年01月05日 星期四 13时53分24秒
                                   
#include <bits/stdc++.h>
#define LL long long
#define fir first
#define sec second
#define pii pair<int,int>
using namespace std;
const int MAXN = 2000000 + 2;
const int P = (int)1e9 + 7;
LL f[MAXN],g[MAXN];
char str[MAXN];
int pre[26];
set<pii> rem;
LL solve(int n,int k){
    int m = strlen(str + 1);
    memset(pre,0,sizeof(pre));
    f[0] = g[0] = 1;
    for(int i=1;i<=m;++i){
        int v = str[i] - 'a';
        if(pre[v] == 0)
            f[i] = g[i - 1];
        else
            f[i] = (g[i - 1] - g[pre[v] - 1] + P) % P;
        g[i] = (g[i - 1] + f[i]) % P;
        pre[v] = i;
//        printf("i = %d f = %lld g = %lld\n",i,f[i],g[i]);
    }
    rem.clear();
    for(int i=0;i<k;++i)
        rem.insert(pii(pre[i],i));
    for(int i=m+1;i<=m+n;++i){
//        puts("ffff");
        pii now = *rem.begin();
        int pos = now.fir,v = now.sec;
        if(pos == 0)
            f[i] = g[i - 1];
        else
            f[i] = (g[i - 1] - g[pos - 1] + P) % P;
        g[i] = (g[i - 1] + f[i]) % P;
        pre[v] = i;
        rem.erase(rem.begin());
        rem.insert(pii(i,v));
    }
//    for(int i=0;i<=m+n;++i)
//        printf("i = %d f = %lld\n",i,f[i]);
    return g[m + n];
}
int main(){
    int n,k;
    scanf("%d %d",&n,&k);
    scanf("%s",str + 1);
    printf("%lld\n",solve(n,k));
    return 0;
}

 

posted on 2017-01-05 21:34  _fukua  阅读(251)  评论(0编辑  收藏  举报