问题描述
小R不再追求甜点中最高的喜爱值,今天他想要的是甜点喜爱值之和正好匹配他的预期值 S。为了达到这个目标,他可以使用魔法棒来改变甜点的喜爱值,使其变为原来喜爱值的阶乘。每个甜点只能使用一次魔法棒,也可以完全不用。
下午茶小哥今天带来了 N 个甜点,每个甜点都有一个固定的喜爱值。小R有 M 个魔法棒,他可以选择任意甜点使用,但每个甜点只能使用一次魔法棒。他的目标是通过选择一些甜点,可能使用魔法棒,使得这些甜点的喜爱值之和恰好为 S。
请计算小R有多少种不同的方案满足他的要求。如果两种方案中,选择的甜点不同,或者使用魔法棒的甜点不同,则视为不同的方案。
测试样例
样例1:
输入:
n = 3, m = 2, s = 6, like = [1, 2, 3]
输出:5
样例2:
输入:
n = 3, m = 1, s = 1, like = [1, 1, 1]
输出:6
样例3:
输入:
n = 5, m = 3, s = 24, like = [1, 2, 3, 4, 5]
输出:1
样例4:
输入:
n = 4, m = 0, s = 10, like = [1, 3, 3, 3]
输出:1
样例5:
输入:
n = 6, m = 1, s = 35, like = [5, 5, 5, 5, 5, 5]
输出:0
通过MarsCode AI进行解题思路分析
问题理解
小R的目标是通过选择一些甜点,可能使用魔法棒,使得这些甜点的喜爱值之和恰好为 S。每个甜点有一个固定的喜爱值,使用魔法棒可以将甜点的喜爱值变为其阶乘。每个甜点只能使用一次魔法棒,也可以完全不用。
数据结构选择
我们可以使用动态规划来解决这个问题。动态规划通常用于解决具有重叠子问题和最优子结构性质的问题。
算法步骤
-
预处理阶乘:首先,我们需要预处理出所有可能的喜爱值的阶乘,因为阶乘计算是一个耗时的操作,预处理可以提高效率。
-
动态规划状态定义:
- 定义一个二维数组
dp[i][j],其中i表示已经使用了i个魔法棒,j表示当前的喜爱值之和。 dp[i][j]表示使用i个魔法棒,使得喜爱值之和为j的方案数。
- 定义一个二维数组
-
状态转移:
- 对于每个甜点,我们可以选择不使用魔法棒,或者使用魔法棒。
- 如果不使用魔法棒,则直接加上当前甜点的喜爱值。
- 如果使用魔法棒,则加上当前甜点的喜爱值的阶乘。
-
初始状态:
dp[0][0] = 1,表示不使用任何魔法棒,喜爱值之和为 0 的方案数为 1。
-
最终结果:
- 最终的结果是所有使用
0到m个魔法棒,使得喜爱值之和为S的方案数之和。
- 最终的结果是所有使用
最后整理得到如下代码
from collections import defaultdict
magic=[1]*100
def pre():
for i in range(1, 100):
magic[i] = magic[i - 1] * i
def solution(n, m, s, like):
pre()
f = defaultdict(int)
f[(0, 0)] = 1
for i in range(1, n + 1):
g = f.copy()
for (a, b), v in g.items():
if b + like[i - 1] <= s:
f[(a, b + like[i - 1])] += v
if a + 1 <= m and b + magic[like[i - 1]] <= s:
f[(a + 1, b + magic[like[i - 1]])] += v
sum_ = 0
for i in range(m + 1):
sum_ += f[(i, s)]
return sum_
if __name__ == "__main__":
print(solution(3, 2, 6, [1,2,3]) == 5 )
print(solution(3, 1, 1, [1,1,1]) == 6 )
总结
通过动态规划,我们可以有效地计算出满足条件的方案数。关键在于正确地定义状态和状态转移方程,以及处理好边界条件。