LeetCode 920. Number of Music Playlists

一看知道是dp题,但是和常规dp很不一样,如何formulate问题很难想。

dp[i][j] 表示最后长度L的list前 i 个中有 j 首unique的歌。dp[L][N] 就是我们的答案。

第一维i很好想。由于题目有要求其他k首歌放过以后才能重复播放歌曲,因此我们要记录其他歌曲的播放情况。因此第二维用至此unique的歌曲数量来表示,可以很简单的解决上述问题。

递推公式两种情况:

如果第i首是迄今为止没有播放过的新歌,那么有 N-(j-1) 首新歌可以选择,dp[i][j] = dp[i-1][j-1] * (N-(j-1))

如果第i首是曾经播放过的歌,根据规则,最近的K首歌都是不能放的,我们有 j-K 种选择,dp[i][j] = dp[i-1][j] * (j-K)

Base case: dp[1][1] = 1,为了方便定义 dp[0][0] = 1,可以把 dp[1][1] 也包括进去。

实际上,只有 dp[i][j],i>=j 的项才是有用的,只更新这些项也是可以的。

class Solution {
public:
    int MOD=1e9+7;
    
    int numMusicPlaylists(int N, int L, int K) {
        vector<vector<long long>> dp(L+1,vector<long long>(N+1,0));
        dp[0][0] = 1;
        for (int i=1;i<=L;++i){
            for (int j=1;j<=N;++j){
                dp[i][j] = (dp[i-1][j-1]*(N-j+1) + dp[i-1][j]*max(j-K,0)) % MOD;
            }
        }
        return dp[L][N];
    }
};

时间复杂度:O(LN)

空间复杂度:O(LN),可以优化为一维。

转载于:https://www.cnblogs.com/hankunyan/p/11470837.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值