蓝桥杯算法——记忆化搜索

!TIP

一种剪枝算法,优化运算效率,减少冗余计算

基本内容

题目要求:输入n,输出一共可以构造多少个数列,要求数列的第 i不能超过第i-1个数的一半

示例:输入6,只能输出 [6], [6, 1], [6, 2], [6, 3], [6, 2, 1], [6, 3, 1] 一共六种

  • 传统思路,深度优先搜索算法,可以发现大部分案例都 TLE(超时)了
python 复制代码
n = int(input())
ans = 0
def f(x):
    global ans
    ans += 1
    for i in range(1, int(x/2)+1):
        f(i)

f(n)
print(ans)
  • 超时分析:存在着重复计算的数列重复子问题,以输入8为例,当我们计算出[8, 2, 1]时就知道了当输入为2时只有俩个可以满足的序列,以此类推,当我们以32为输入,计算到[36, 16, 8 ...] 得知子树8共有 10 种数列时,即可直接计算 [36, 8] 共有11种满足的数列。
  • 记忆化搜索 :额外开辟一个数组空间 cache 存储计算过的值
python 复制代码
n = int(input())
cache = [-1] * (n+1)
def f(x):
    if cache[x] != -1: # cache 不为-1表示已经计算过
        return cache[x]
    
    ans = 1 # 每一个数字都可以表示单独为一个数列
    for i in range(1, int(x/2)+1):
        ans += f(i)
    cache[x] = ans
    return ans

print(f(n))
题目
  1. Leetcode 638. 大礼包

本来想用贪心算法去做,行不通,还是需要遍历每一种情况

python 复制代码
class Solution:
    def shoppingOffers(self, price: List[int], special: List[List[int]], needs: List[int]) -> int:
        cache = {}

        def dfs(needs: Tuple[int]) -> int:
            if needs in cache: # 若need被计算过则返回need所需的最小花费
                return cache[needs]
            
            min_cost = 0 # 不用礼包的低消
            for i in range(len(price)):
                min_cost += (needs[i] * price[i])
          
            for offer in special: # 因为礼包可以无限次使用,每次都需要遍历每一个礼包
                new_needs = []
                for i in range(len(needs)):
                    if needs[i] < offer[i]:  # 如果大礼包的物品超出需求,跳过
                        break
                    new_needs.append(needs[i] - offer[i])
                else: # 表示for循环没有被break,计算当前使用该礼包是否可以得到最小值
                    min_cost = min(min_cost, dfs(tuple(new_needs)) + offer[-1])
                    
            cache[needs] = min_cost
            return min_cost
        
        return dfs(tuple(needs))
  1. Leetcode 552. 学生出勤记录 II

!note

@cache和字典存储的方式一样,都是保存某一状态的值,但是用字典存储的话有可能会超出内存限制,@cache不会,因为会自动回收状态,如果使用字典的话需要定时清楚字典的存储空间。

  • 字典存储(超出内存限制
python 复制代码
class Solution:
    def checkRecord(self, n: int) -> int:
        def dfs(i, j, k):
            if i >= n:
                return 1
            if (i, j, k) in memo:
                return memo[(i, j, k)] 

            ans = 0
            if j == 0:
                ans += dfs(i + 1, j + 1, 0)
            if k < 2:
                ans += dfs(i + 1, j, k + 1)
            ans += dfs(i + 1, j, 0)
            
            memo[(i, j, k)] = ans % mod  
            return memo[(i, j, k)]
        mod = 10**9 + 7
        memo = {}
        return dfs(0, 0, 0)
  • @cache
python 复制代码
class Solution:
    def checkRecord(self, n: int) -> int:
        @cache
        def dfs(i, j, k):
            if i >= n:
                return 1
            ans = 0
            if j == 0:
                ans += dfs(i + 1, j + 1, 0)
            if k < 2:
                ans += dfs(i + 1, j, k + 1)
            ans += dfs(i + 1, j, 0)
            return ans % mod

        mod = 10**9 + 7
        ans = dfs(0, 0, 0)
        dfs.cache_clear()
        return ans
相关推荐
旖-旎4 分钟前
哈希表(存在重复元素||)(4)
数据结构·c++·算法·leetcode·哈希算法·散列表
Run_Teenage6 分钟前
Linux:认识信号,理解信号的产生和处理
linux·运维·算法
無限進步D21 分钟前
蓝桥杯赛前刷题
c++·算法·蓝桥杯·竞赛
CoderCodingNo23 分钟前
【GESP】C++二级真题 luogu-B4497, [GESP202603 二级] 数数
开发语言·c++·算法
磊 子26 分钟前
八大排序之冒泡排序+选择排序
数据结构·算法·排序算法
We་ct27 分钟前
LeetCode 50. Pow(x, n):从暴力法到快速幂的优化之路
开发语言·前端·javascript·算法·leetcode·typescript·
潇洒畅想32 分钟前
1.1 从∑到∫:用循环理解求和与累积
java·数据结构·python·算法
郝学胜-神的一滴1 小时前
[简化版 GAMES 101] 计算机图形学 04:二维变换上
c++·算法·unity·godot·图形渲染·unreal engine·cesium
ZC跨境爬虫1 小时前
海南大学交友平台开发实战day7(实现核心匹配算法+解决JSON请求报错问题)
前端·python·算法·html·json
计算机安禾1 小时前
【数据结构与算法】第41篇:图论(五):拓扑排序与关键路径
c语言·数据结构·c++·算法·图论·visual studio