【每日一题】Removal(4.27)
Removal
https://ac.nowcoder.com/acm/problem/17137
dp计数问题
先考虑简单一点的情况,如果允许重复的话,可以用dp[i][j]表示当前位置为i,并且删除了j个元素后的方法数,那么转移相当于考虑第i个位置的数选和没选两种情况,即dp[i][j] = dp[i-1][j] + dp[i-1][j-1],从i=1到n扫描一遍就可以得到答案了。
这道题要求不重复,如果用dp[i][j]表示当前位置为i,删除了j个元素后不重复的方法数,怎么转移呢?
如果还用dp[i][j] = dp[i-1][j-1]可能会引起重复,思考下引起重复的原因肯定是第i个位置的数选中,
那么我们只需要找到上一个a[i]出现的位置,以a[i]结尾并且子串的长度和i-j相等的就是和dp[i][j]重复的,直接减去即可.
#include <iostream>
#include <cstring>
using namespace std;
const int maxn = 1e5+7;
const int mod = 1e9+7;
int a[maxn],dp[maxn][11],l[11]; //l记录上一次出现的位置
int main()
{
ios::sync_with_stdio(0);cin.tie(0);
int n,m,k;
while(cin>>n>>m>>k)
{
for(int i=1;i<=n;i++) cin>>a[i];
for(int i=0;i<=11;i++) dp[i][i] = 1; //全删除
for(int i=1;i<=n;i++) dp[i][0] = 1; //删除0个元素
memset(l,0,sizeof(l));
for(int i=1;i<=n;i++)
{
for(int j=1;j<=min(i-1,m);j++)
{
dp[i][j] = (dp[i-1][j-1] + dp[i-1][j])%mod;
int len = i-j; //字串的长度
if(l[a[i]])
{
int pos = l[a[i]];
int cnt = pos -len; //要删除的元素的个数
if(cnt>=0)
dp[i][j] = (dp[i][j]-dp[pos-1][cnt]+mod)%mod;
}
}
l[a[i]] = i;
}
cout<<dp[n][m]<<'\n';
}
return 0;
}