【每日一题】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;
}

全部评论

相关推荐

程序员牛肉:1.大头肯定是院校问题,这个没啥说的。 2.虽然有实习,但是实习的内容太水了,在公司待了七个月的时间,看起来就只做了jwt和接入redis。爬取新闻,数据导入。这几个需求值得你做七个月吗?这不就是三四个月的工作量吗?我要是面试官的话真心会认为你能力不太行。所以既然有实习了,一定要好好写,像是Swagger这种东西是真没必要写上去,就拉一个包的事情。 3.我个人觉得话,在校生不要把自己当社招看,除非你的项目是特别牛逼,特别有名的含金量,否则不要写这种密密麻麻的一串子工作职责。你的项目只有一个作用,就是供面试官从中来抽取八股对你进行拷打。 但是你现在这个看不来什么技术点,可以改一下,详细表述一下你用什么技术实现了什么功能,在实现这个功能的过程中,你解决了什么难题。
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务