2021.05.03播放列表的数量

2021.05.03播放列表的数量

题目描述

你的音乐播放器里有 N 首不同的歌,在旅途中,你的旅伴想要听 L 首歌(不一定不同,即,允许歌曲重复)。请你为她按如下规则创建一个播放列表:

  1. 每首歌至少播放一次。
  2. 一首歌只有在其他 K 首歌播放完之后才能再次播放。

返回可以满足要求的播放列表的数量。由于答案可能非常大,请返回它模 10^9 + 7 的结果。

数据规模和约定

0 <= K < N <= L <= 100

思路

  1. 本次动态规划和之前不同的是动态规划大环境的设定不同。由题目中给出的N,L,K设定动态规划的大环境,而其中的各个状态是在该环境下的不同情况。
  2. 如在给定的NLK情况下,已经排列了i首歌曲,其中有j首歌曲是不同的。并且目前排列好的歌曲满足K的要求。 --dp[i][j]
  3. 状态转移方程:如代码所示:

代码

	public int numMusicPlaylists(int N, int L, int K) {
		int MOD = 1_000_000_007;
		long[][] dp = new long[L+1][N+1];
		dp[0][0] = 1; //初始化条件
		for(int i = 1; i <= L; i++) {
			for(int j = 1; j <= i && j <= N; j++) {
				//如果新安排的第i首歌曲是之前没有的,那么找dp[i-1][j-1],共可以安排N-(j-1)种歌曲。
				dp[i][j] += dp[i-1][j-1]*(N-(j-1)); 
				//如果新安排的第i首歌曲是之前安排过的,那么找dp[i-1][j],共可以安排j-k种(因为在第i首之前的k首不能安排,防止重复)
				dp[i][j] += dp[i-1 ][j]*Math.max(j-K, 0);
				//防止数值过大
				dp[i][j] %= MOD;
			}
		}
		return (int)dp[L][N];
    }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值