[UOJ#149][NOIP2015]子串(dp)

题目描述

传送门

题解

设f(i,j,k)表示从a的前j个字符中选i段连接起来可以和b的前k个匹配的方案数。可以预处理出来g(i,j)表示a的第i个字符和b的第j个字符从后往前最多能匹配多少个。
那么 f(i,j,k)=l=1g(i,j)f(i1,jl,kl)
可以肯定的是dp是三维状态的,而且是 O(n) 转移,时间和空间都是爆掉的。考虑怎么优化。
其实优化都比较简单。空间上可以发现f(i)只与f(i-1)有关,可以用滚动数组。时间上发现转移的时候是加了一段连续的和,可以使用前缀和优化。如果把f(i)中的两维看成一个二维平面的话,这里的前缀和应该是点(x,y)一直向左上角走直到不能走为止所经过的点的和。
时间复杂度 O(nmk)

代码

#include
#include
#include
using namespace std;
#define N 1005
#define M 205
#define Mod 1000000007

int n,m,p;
int f[2][N][M],s[2][N][M],g[N][M];
char a[N],b[M];

void init()
{
    for (int i=1;i<=n;++i)
        for (int j=1;j<=m;++j)
        {
            g[i][j]=min(i,j);
            for (int k=1;k<=min(i,j);++k)
                if (a[i-k+1]!=b[j-k+1])
                {
                    g[i][j]=k-1;
                    break;
                }
        }
}
int main()
{
    scanf("%d%d%d\n",&n,&m,&p);
    gets(a+1);gets(b+1);

    init();
    for (int i=1;i<=m;++i)
        for (int j=i;j<=n;++j)
        {
            f[1][j][i]=f[1][j-1][i];
            if (g[j][i]==i) f[1][j][i]++;
            s[1][j][i]=s[1][j-1][i-1]+f[1][j][i];
        }

    for (int i=2;i<=p;++i)
    {
        memset(f[i&1],0,sizeof(f[i&1]));
        memset(s[i&1],0,sizeof(s[i&1]));
        for (int j=1;j<=n;++j)
            for (int k=1;k<=m;++k)
            {
                f[i&1][j][k]=f[i&1][j-1][k];
                if (g[j][k])
                {
                    int x=s[(i-1)&1][j-1][k-1];
                    f[i&1][j][k]=(f[i&1][j][k]+s[(i-1)&1][j-1][k-1])%Mod;
                    int J=max(j-g[j][k]-1,0),K=max(k-g[j][k]-1,0);
                    int y=s[(i-1)&1][J][K];
                    f[i&1][j][k]=((f[i&1][j][k]-s[(i-1)&1][J][K])%Mod+Mod)%Mod;
                }
                s[i&1][j][k]=(s[i&1][j-1][k-1]+f[i&1][j][k])%Mod;       
            }
    }
    printf("%d\n",f[p&1][n][m]);
}

你可能感兴趣的:(题解,dp,NOIP)