本文详解在实现 `how_sum` 递归函数时,因误用可变默认参数(`memo={}`)导致缓存污染、结果错误的问题,并提供安全、高效、可复用的记忆化解决方案。
在解决“寻找数组中若干元素使其和等于目标值”这类经典组合问题时,朴素递归虽逻辑清晰,但存在大量重复子问题,时间复杂度呈指数级增长(如 O(n^target))。引入记忆化(memoization)是标准优化手段——但错误的缓存设计反而会引发隐蔽且严重的逻辑错误,正如本例所示。
原始代码中使用了:
def how_sum(target, nums, memo: dict = {}) -> ...Python 中,默认参数在函数定义时仅初始化一次,而非每次调用时新建。这意味着:
⚠️ 这正是 memo[target] = rv + [num] 一行引发错误的核心原因:它向被跨调用共享的 memo 中写入了依赖于特定 nums 的结果,破坏了缓存的上下文隔离性。
安全的记忆化必须保证:每个独立的顶层调用(即不同 target 和 nums 组合)拥有专属、隔离的缓存空间。推荐做法是:
以下是修复后的完整实现:
def how_sum(
target: int,
nums: list[int],
memo: dict[int, list[int] | None] | None = None
) -> list[int] | None:
# ✅ 每次调用确保 memo 是独立实例
if memo is None:
memo = {0: []} # base case: sum 0 → empty list
# ✅ 缓存命中直接返回
if target in memo:
return memo[target]
# ✅ 剪枝:负数无解
if target < 0:
memo[target] = None
return None
# ✅ 遍历所有候选数字
for num in nums:
remainder = target - num
combination = how_sum(remainder, nums, memo)
# ✅ 找到有效组
合:拼接并缓存
if combination is not None:
result = combination + [num]
memo[target] = result
return result
# ✅ 当前 target 无解,缓存 None 并返回
memo[target] = None
return Nonedef main():
print(how_sum(7, [2, 3])) # [3, 2, 2] 或 [2, 2, 3](顺序取决于遍历)
print(how_sum(7, [5, 3, 4, 7])) # [7] 或 [4, 3](任一合法解均可)
print(how_sum(7, [2, 4])) # None(2 和 4 均为偶数,无法组成奇数 7)
print(how_sum(8, [2, 3, 5])) # [3, 5] 或 [2, 2, 2, 2]
print(how_sum(500, [7, 14])) # None(所有组合均为 7 的倍数,500 不是)
if __name__ == "__main__":
main()? 关键提示:该函数返回「任意一个可行解」,不保证最短/字典序最小。若需最优解(如最少元素),应改用动态规划并记录路径,或在递归中比较各分支长度。
通过修正这一细微却关键的设计缺陷,你的 how_sum 函数即可在保持简洁递归结构的同时,获得接近线性的时间效率,稳健应对大规模输入。