洛谷P2679 子串 dp

正解:dp+滚动数组(...是叫这个名儿趴qwq

解题报告:

感觉是道dp好题啊,所以就写了个题解

代码实现难度低,思维难度大,像我这种思维僵化傻逼选手只想到了爆搜+组合数学...

其实是道很妙的dp题!好趴也没有多妙主要大概是妙在想到了dp?具体实现很普通的

然后代码没什么要解释的鸭,真的不难,主要觉得题目出得还是挺好的趴,像我这种傻逼根本想不到是dp的鸭...然后实在是想放上来就放上来了

具体实现随便说点儿趴

就f[i][j][k]:A串扫到i B串扫到j 有k个空格 然后这里会发现有俩问题

1)因为f存的是个和我并不知道有多少是上一位匹配上的也就不知道上一个相等的那个位置和这一位之间有没有断电

2)ijl乘起来太大了会MLE掉

解决方法对应也是俩

1)再开个辅助数组g[i][j][k]表示是正儿八经这一位匹配上的

2)可以发现每次转移时i只和上一个i有关,滚动数组处理掉就好

over

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define rp(i,x,y) for(register ll i=x;i<=y;++i)
#define my(i,x,y) for(register ll i=x;i>=y;--i)

const ll mod=1000000007;
ll n,m,cjk,f[2][210][210],g[2][210][210];
bool now=1;
string a,b;

inline ll read()
{
    char ch=getchar();ll x=0;bool y=1;
    while(ch!='-' && (ch>'9' || ch<'0'))ch=getchar();
    if(ch=='0')y=0,ch=getchar();
    while(ch>='0' && ch<='9')x=(x<<1)+(x<<3)+(ch^'0'),ch=getchar();
    return y?x:-x;
}

int main()
{
    string s;
    n=read();m=read();cjk=read();cin>>a>>b;f[0][0][0]=1;a=' '+a;b=' '+b;
    rp(i,1,n)
    {
        f[now][0][0]=1;
        rp(j,1,m)
        {
            rp(k,1,cjk)
            {
                if(a[i]==b[j])g[now][j][k]=(f[now^1][j-1][k-1]+g[now^1][j-1][k])%mod;//g:可以从之前的f转移来,就必有空格本来没空格也强行当有 也可以从上一位的g转移来,就中间没有空格
                else g[now][j][k]=0;
                f[now][j][k]=(f[now^1][j][k]+g[now][j][k])%mod;
            }
        }
        now^=1;
    }
    printf("%lld\n",f[now^1][m][cjk]);
    return 0;
}
反正我就是觉得这题很好qwq

猜你喜欢

转载自www.cnblogs.com/lqsukida/p/9910915.html