【每日一题】Removal (dp+思维 / 子序列 计数)

Removal

https://ac.nowcoder.com/acm/problem/17137

Solution
题意:给出n个元素,每个元素不大于k,求 删除m个元素后的子序列个数。

子序列问题,通常可以联想到dp来做,
考虑 维护 前 i 个元素删除 j 个元素的方案数,
考虑第 i 个元素删或者不删,即有:

但是这样计算的话肯定会有重复的方案,
拿样例2来说:
4 2 2
1 2 1 2
删除第一个元素和删除第二个元素 得到 1 2
删除第三个元素和删除第四个元素 得到 1 2
删除第二个元素和删除第三个元素 也得到 1 2
这样就重复了。

如何计算重复的子序列?
就像求路径一样,如果经过同一个点两次可以说明这条路已经走过了。
那么如果 a[i]在 i 之前出现过,设距离 i 最近的一个下标为x,
就有 a[x]+ i 之后某个序列 = a[i] + i 之后某个序列
重复的方案为 以 x 为右区间的且 x 不删除 且 删除 j-(i-x) 个元素 的方案数
即:
而当如果 i-x 之间的数比 j 大,就不会出现重复的情况
所以考虑用 last[i] 维护上一个 a[i] 出现的位置,也就是上文的x,枚举的时候判断然后减去重复的方案即可:

Code

#include<bits/stdc++.h>
#define mp make_pair
#define pb push_back
#define ll long long
#define fi first
#define se second
#define inf 0x3f3f3f3f
#define io std::ios::sync_with_stdio(false),cin.tie(0),cout.tie(0)
using namespace std;
inline ll read(){ll s=0,w=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}
while(ch>='0'&&ch<='9') s=s*10+ch-'0',ch=getchar();return s*w;}
void put1(){ puts("Yes") ;}void put2(){ puts("No") ;}void put3(){ puts("-1"); }
ll qp(ll a,ll b, ll p){ll ans = 1;while(b){if(b&1){ans = (ans*a)%p;--b;}a =
(a*a)%p;b >>= 1;}return ans%p;}
const int mo=998244353; const int mod=1000000007;

const int manx=1e5+5;

ll dp[manx][20],a[manx],last[manx];
map<ll,ll>vis;

int main(){
    ll n,m,k;
    io;
    while(cin>>n>>m>>k){
        for(int i=1;i<=n;i++) cin>>a[i],last[i]=vis[a[i]],vis[a[i]]=i;
        for(int i=0;i<=n;i++) 
            if(i<m) dp[i][0]=dp[i][i]=1;
            else dp[i][0]=dp[i][m]=1;
        for(int i=1;i<=n;i++){
            ll k=min(i-1ll,m);
            for(int j=1;j<=k;j++){
                dp[i][j]=dp[i-1][j]+dp[i-1][j-1];
                dp[i][j]%=mod;
                if(last[i]&&j>=i-last[i])
                    dp[i][j]-=dp[last[i]-1][j-(i-last[i])];
                dp[i][j]=(dp[i][j]+mod)%mod;
            }
        }
        cout<<dp[n][m]<<endl;
        for(int i=1;i<=n;i++) last[i]=0;  vis.clear();
    }
    return 0;
}
全部评论

相关推荐

评论
点赞
收藏
分享

创作者周榜

更多
正在热议
更多
# 春招至今,你的战绩如何? #
9578次浏览 87人参与
# 你的实习产出是真实的还是包装的? #
1710次浏览 40人参与
# MiniMax求职进展汇总 #
23832次浏览 308人参与
# 军工所铁饭碗 vs 互联网高薪资,你会选谁 #
7453次浏览 43人参与
# 简历第一个项目做什么 #
31555次浏览 330人参与
# 重来一次,我还会选择这个专业吗 #
433352次浏览 3926人参与
# 巨人网络春招 #
11301次浏览 223人参与
# 当下环境,你会继续卷互联网,还是看其他行业机会 #
186969次浏览 1122人参与
# 牛客AI文生图 #
21408次浏览 238人参与
# 不考虑薪资和职业,你最想做什么工作呢? #
152290次浏览 887人参与
# 研究所笔面经互助 #
118872次浏览 577人参与
# 简历中的项目经历要怎么写? #
310060次浏览 4193人参与
# AI时代,哪些岗位最容易被淘汰 #
63407次浏览 803人参与
# 面试紧张时你会有什么表现? #
30488次浏览 188人参与
# 你今年的平均薪资是多少? #
213009次浏览 1039人参与
# 你怎么看待AI面试 #
179861次浏览 1234人参与
# 高学历就一定能找到好工作吗? #
64313次浏览 620人参与
# 你最满意的offer薪资是哪家公司? #
76436次浏览 374人参与
# 我的求职精神状态 #
447984次浏览 3128人参与
# 正在春招的你,也参与了去年秋招吗? #
363243次浏览 2637人参与
# 腾讯音乐求职进展汇总 #
160584次浏览 1111人参与
# 校招笔试 #
470425次浏览 2963人参与
牛客网
牛客网在线编程
牛客网题解
牛客企业服务