CF1729G

Problem - 1729G - Codeforces

一道很妙的计数DP。

对于样例一:

ababa bacababa
aba

对于ababa,我们可以删除3位置或5位置。

那么思考何时不用删5位置?显然3位置被删除之后,5位置不用进行删除。

所以现在 i 位置是匹配的位置,当区间[ i-m+1,i-1 ] (m为T的长度 )有一个位置被删了,i位置就可以不用删。

否则i位置必须删除

故可以考虑:dp状态dp[i][0/1]表示:前 i 位已经是合法情况,i位置不删还是删情况下的方案数。

但是这依然无法解决最少的删除次数这一问题。注意到题目无论 |S| 还是 |T|都挺小的,所以我们进一步dp状态。 

 dp状态dp[i][j][0/1]表示:前 i 位已经是合法情况,此时进行了j次删除操作,i位置不删还是删情况下的方案数。(这个dp状态便是此题的精髓)

接下来是dp转移:

一:i 位置不是匹配位置:dp[i][j][0]=dp[i-1][j][0]+dp[i-1][j][1](前面是删还是不删不知道,所以都加)

二:i 位置是匹配位置:dp[i][j][0]=∑dp[x][j][1]  x∈[ i-m+1,i-1 ],j>0

            dp[i][j][1]=dp[i-m][j-1][0]+dp[i-m][j-1][1]  j>0

对于∑dp[x][j][1] x∈[ i-m+1,i-1 ],我们可以多开个sum数组来优化时间复杂度。

则 i 位置是匹配位置:dp[i][j][0]=sum[i-1][j]-sum[i-m][j]

最后考虑:最少的删除次数。根据我们dp状态,dp[n][i][0]+dp[n][i][1]>0时,i 便是可行的删除次数,所以找到第一个i符合方案数>0即可

初始化:dp[0][0][0]=1,注意dp[0][0][1]依旧为0

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define mp make_pair
#define pb push_back   
#define popb pop_back  
#define fi first
#define se second
#define popcount __builtin_popcount
const int N=510;
const ll MOD=1e9+7;
const int inf=1e9;
//const ll INF=1e18;
int T,n,m;
bool pos[N];
ll dp[N][N][2],sum[N][N];
string s,t;
inline int read()
{
    int x=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=(x<<1)+(x<<3)+(ch^48);ch=getchar();}
    return x*f;
}
int main()
{
//    freopen("","r",stdin);
//    freopen("","w",stdout);
    ios::sync_with_stdio(0);
    cin.tie(0);
    cin>>T;
    while(T--)
    {
        cin>>s>>t;
        n=s.size();
        m=t.size();
        s='?'+s;
        for(int i=1;i<=n;i++)
            if(s.substr(i,m)==t) pos[i+m-1]=1;
        dp[0][0][0]=1;
        for(int i=1;i<=n;i++)
            for(int j=0;j<=n;j++)
            {
                if(pos[i]) 
                {
                    if(j) dp[i][j][1]=(dp[i-m][j-1][0]+dp[i-m][j-1][1])%MOD;
                    if(j) dp[i][j][0]=(sum[i-1][j]-sum[i-m][j]+MOD)%MOD;
                }
                else dp[i][j][0]=(dp[i-1][j][0]+dp[i-1][j][1])%MOD;
                sum[i][j]=(dp[i][j][1]+sum[i-1][j])%MOD;
            }
        for(int i=0;i<=n;i++)
        {
            if(dp[n][i][0]+dp[n][i][1])
            {
//                printf("%d %lld\n",i,(dp[n][i][0]+dp[n][i][1])%MOD);
                cout<<i<<" "<<(dp[n][i][0]+dp[n][i][1])%MOD<<"\n";
                break;
            }
        }
        for(int i=1;i<=n;i++) pos[i]=0;
        for(int i=1;i<=n;i++)
            for(int j=0;j<=n;j++)
                dp[i][j][0]=dp[i][j][1]=sum[i][j]=0;
    } 
    return 0;
}
// if(pos[i]) dp[i][j][1]=dp[i-m][j-1][0]+dp[i-m][j-1][1]
// if(pos[i]) dp[i][j][0]=sum(dp[k][j][1]) k∈[i-m+1,i-1]
// else dp[i][j][0]=dp[i-1][j][0]+dp[i-1][j][0] 

 

posted @ 2023-04-29 21:26  QAQ啥也不会  阅读(12)  评论(0编辑  收藏  举报