CodeForces - 1183H Subsequences (hard version) (DP)

题目:https://vjudge.net/contest/325352#problem/Cc++

题意:输入n,m,给你一个长度为n的串,而后你有一个集合,集合里面都是你的子序列,集合里面不能重复,集合中元素的花费是 n-当前元素长度 ,也就是删除了几个字符,而后要你求前m个最小花费是多少spa

思路:咱们考虑dp,dp[i][j] 前i个字符删除j个字符的方案数,咱们先假设没有重复字符,没有的话,很明显转移方程就是dp[i][j]=dp[i-1][j-1]+dp[i-1][j].net

也就是考虑   当前字符删+当前字符不删  ,可是若是有重复的字符咱们怎么处理呢,若是abcda,咱们遇到第一个a,后面再遇到 第二个a时,咱们能够删掉code

abcd和 bcda 结果都是a,咱们怎么处理呢,发现咱们想获得相同的串,咱们确定要把两个字符之间的位置字符全删了,而后再删其中一边,全部咱们能够考虑先把中间删掉,而后把重复那一部分,也就是前一个出现位置的串数删掉便可,最后咱们再从删掉字符数从低到高枚举便可blog

 

 

#include<bits/stdc++.h>
#define maxn 100005
#define mod 1000000007
using namespace std;
typedef long long ll;
ll n,m,dp[105][105];
char str[maxn];
int main(){
    scanf("%lld%lld",&n,&m);
    scanf("%s",str+1);
    dp[0][0]=1;
    for(int i=1;i<=n;i++){
        dp[i][0]=1;
        for(int j=1;j<=i;j++){
            dp[i][j]=dp[i-1][j-1]+dp[i-1][j]; 
            for(int k=i-1;k>=1&&(i-k)<=j;k--){
                if(str[i]==str[k]){
                    dp[i][j]-=dp[k-1][j-(i-k)];
                    break;
                }    
            }
        }
    }
    ll sum=0;
    for(int i=0;i<=n;i++){
        if(m>=dp[n][i]){
            sum+=i*dp[n][i];
            m-=dp[n][i];
        }
        else{
            sum+=i*m;
            m=0;
            break;
        }
    }
    if(m>0) cout<<"-1";
    else cout<<sum;
} 
相关文章
相关标签/搜索