记忆化搜索(5题)

是什么? 是一个带备忘录的递归

如何实现记忆化搜索

1.添加一个备忘录(建立一个可变参数和返回值的映射关系)

2.递归每次返回的时候把结果放到备忘录里

3.在每次进入递归的时候往备忘录里面看看。

目录

1.斐波那契数列

2.不同路径

3.最长递增子序列

4.猜数字大小2

5.矩阵中最长的递增路径


1.斐波那契数列

LCR 126. 斐波那契数 - 力扣(LeetCode)

         我们再看看动态规划的做法。我们发现其实两者具有一一对应的关系,只不过一种是用递归形式呈现,另一种是用递推形式呈现

小问题:

1. 所有的递归(暴搜、深搜),都能改成记忆化搜索吗?

不是的,只有在递归的过程中,出现了大量完全相同的问题时,才能用记忆化搜索的方式优化

2.带备忘录的递归vs动态规划 vs记忆化搜索

本质上是一回事

3.自顶向下  vs  自低向上

不同的看待问题的方式

4.暴搜->记忆化搜索->动态规划

可以为我们的动态规划提供一个方向

        我们创建一个备忘录的时候需要在里面填入一个不可能取到的值,以此来判断备忘录是否已经更新

class Solution {
public:
    int memo[110];//一个备忘录
    int fib(int n) {
        memset(memo, -1, sizeof(memo));
        return dfs(n);
    }
    int dfs(int n)
    {
        if(memo[n] != -1)return memo[n];//每次进入的时候查看备忘录里面是否存值
        if(n == 1)
        {
            memo[1] = 1;
            return memo[1];//返回之前把值存到备忘录里面
        }
        else if(n == 0)
        {
            memo[0] = 0;
            return memo[0];
        }
        else 
        {
            memo[n] = (dfs(n-1) + dfs(n - 2))%1000000007;
            return memo[n];
        }
    }
};

2.不同路径

62. 不同路径 - 力扣(LeetCode)

1.暴力搜索(会超时)

class Solution {
public:
    int uniquePaths(int m, int n) {
        return dfs(m,n);
    }
    int dfs(int i, int j)
    {
        if(i == 0 || j == 0)return 0;
        if(i == 1 && j == 1)return 1;
        return dfs(i - 1, j) + dfs(i, j - 1);
    }
};

2.记忆化搜索

class Solution {
public:
    int uniquePaths(int m, int n) {
        vector<vector<int>> memo(m +1, vector<int>(n + 1));
        for(int i = 0; i < m + 1; i++)
        {
            for(int j = 0; j < n + 1; j++)
            {
                memo[i][j] = -1;
            }
        }

        return dfs(m,n,memo);
    }
    int dfs(int i, int j, vector<vector<int>> & memo)
    {
        if(memo[i][j] != -1)return memo[i][j];
        if(i == 0 || j == 0)
        {
            return 0;
        }
        if(i == 1 && j == 1)
        {
            memo[1][1] = 1;
            return memo[1][1];
        }
        memo[i][j] = dfs(i - 1, j,memo) + dfs(i, j - 1,memo);
        return memo[i][j];
    }
};

3.最长递增子序列

 300. 最长递增子序列 - 力扣(LeetCode)

1.暴搜(会超时) 

dfs(pos)的任务是返回以nums[pos]为开头的最长子序列

我们在主函数里面从头开始遍历一次,找到最大的那个,ret 一开始为0.

在dfs函数中,ret一开始为1,因为已经进入dfs函数的情况下最少也有一个数。

我们从pos的下一位开始,遍历到结尾,每遍历到一个位置我们判断这个值是否比nums【i】大,如果更大说明它可以连接到nums【i】后面,我们再更新ret, ret = max(ret, dfs(nums, i) + 1);

class Solution {
public:
    int n;
    int lengthOfLIS(vector<int>& nums) {
        int ret = 0;
        n = nums.size();
        for(int i = 0; i < n; i++)
        {
            ret = max(dfs(nums, i), ret);
        }
        return ret;
    }
    int dfs(vector<int>& nums, int pos)
    {
        int ret = 1;
        for(int i = pos + 1; i < n; i++)
        {
            if(nums[i] > nums[pos])
            {
                ret = max(ret, dfs(nums, i) + 1);
            }
        }
        return ret;
    }
};

2.记忆化搜索

记录数据需要在返回之前记录。

class Solution {
public:
    int n;
    int lengthOfLIS(vector<int>& nums) {
        int ret = 0;
        n = nums.size();
        vector<int> memo(n);
        for(int i = 0; i < n; i++)
        {
            ret = max(dfs(nums, i,memo), ret);
        }
        return ret;
    }
    int dfs(vector<int>& nums, int pos,vector<int>& memo)
    {
        if(memo[pos] != 0)return memo[pos];
        int ret = 1;
        for(int i = pos + 1; i < n; i++)
        {
            if(nums[i] > nums[pos])
            {
                ret = max(ret, dfs(nums, i, memo) + 1);
            }
        }
        memo[pos] = ret;
        return ret;
    }
};

3.递归代码

dp[i]表示的是从i下标开始最大的一个子序列。

因此我们从末尾开始填dp表,每个位置最小的子序列是自身,因此dp表每个位置先赋值为1.

 每次进行比较之后才更新

 if(nums[i] < nums[j])
                {
                    dp[i] = max(dp[i], dp[j] + 1);
                }

最后的结果从dp表里面挑一个最大的即可。

class Solution {
public:
    int lengthOfLIS(vector<int>& nums) {
        int n = nums.size();
        int ret = 0;
        vector<int> dp(n,1);
        for(int i = n - 1; i >= 0; i--)
        {
            for(int j = i + 1; j < n; j++)
            {
                if(nums[i] < nums[j])
                {
                    dp[i] = max(dp[i], dp[j] + 1);
                }
            }
            ret = max(ret, dp[i]);
        }
        return ret;
    }
};

4.猜数字大小2

375. 猜数字大小 II - 力扣(LeetCode)

1.暴搜(会超时)

这题让我们求  不管猜的是哪个数字 确保获胜  的最小金额

        首先,猜的策略有很多种,我们用暴力搜索,从小到大依次以某个值为根,然后分出两个树,继续对两个树进行相同的操作。

我们要确保无论它让我们猜哪个值都要获胜,那么就应该找每棵树里面花钱最多的策略,

我们使得int l = dfs(left,i - 1);
            int r = dfs(i + 1, right);

花钱最多的 即根节点+ 左右两树中花钱多的即 i + max(l, r) , 

同时又要最少金额,那么就从各种一定会获胜的金额中取出最小的即可

class Solution {
public:
    int getMoneyAmount(int n) {
        return dfs(1, n);
    }
    int dfs(int left, int right)
    {
        if(left >= right)return 0;
        int ret = INT_MAX;
        for(int i = left; i <= right; i++)
        {
            int l = dfs(left,i - 1);
            int r = dfs(i + 1, right);
            ret = min(ret, i + max(l, r));
        }
        return ret;
    }
};

2.记忆化搜索

class Solution {
public:
    int getMoneyAmount(int n) {
        vector<vector<int>> memo(n + 1, vector<int>(n + 1, -1));
        return dfs(1, n, memo);
    }
    int dfs(int left, int right, vector<vector<int>> & memo)
    {
        if(left >= right)return 0;
          if(memo[left][right] != -1)return memo[left][right];
        int ret = INT_MAX;
        for(int i = left; i <= right; i++)
        {
            int l = dfs(left,i - 1, memo);
            int r = dfs(i + 1, right, memo);
            ret = min(ret, i + max(l, r));
        }
        memo[left][right] = ret;
        return ret;
    }
};

5.矩阵中最长的递增路径

 LCR 112. 矩阵中的最长递增路径 - 力扣(LeetCode)

 

class Solution {
public:
    //int check[210][210];
    int m,n;
    int dx[4] = {0 , -1, 0, 1};
    int dy[4] = {-1, 0, 1, 0};
    int longestIncreasingPath(vector<vector<int>>& matrix) {
        m = matrix.size();
        n = matrix[0].size();
        vector<vector<int>> memo(m, vector<int>(n, -1));
        int ret = 0;
        for(int i = 0; i < m; i++)
        {
            for(int j = 0; j < n; j++)
            {
                ret = max(ret,dfs(i,j,matrix,memo));
            }
        }
        return ret;
    }
    int dfs(int x, int y, vector<vector<int>>& matrix, vector<vector<int>>& memo)
    {
        if(memo[x][y] != -1)return memo[x][y];
        int ret = 1;
        for(int k = 0; k < 4; k++)
        {
            int i = x + dx[k];
            int j = y + dy[k];
            if(i >= 0 && i < m && j >= 0 && j < n )
            {
                if(matrix[i][j] > matrix[x][y])
                {
                    ret = max(ret, dfs(i,j,matrix,memo) + 1);
                }
            }
        }
        memo[x][y] = ret;
        return ret;
    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/962410.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Redis_Redission的入门案例、多主案例搭建、分布式锁进行加锁、解锁底层源码解析

目录 ①. Redis为什么选择单线程&#xff1f; ②. 既然单线程这么好,为什么逐渐又加入了多线程特性&#xff1f; ③. redis6的多线程和IO多路复用入门篇 ④. Redis6.0默认是否开启了多线程&#xff1f; ⑤. REDIS多线程引入总结 ①. Redis为什么选择单线程&#xff1f; ①…

本地运行大模型效果及配置展示

电脑上用ollama安装了qwen2.5:32b&#xff0c;deepseek-r1:32b&#xff0c;deepseek-r1:14b&#xff0c;llama3.1:8b四个模型&#xff0c;都是Q4_K_M量化版。 运行过程中主要是cpu和内存负载比较大&#xff0c;qwen2.5:32b大概需要22g&#xff0c;deepseek-r1&#xff1a;32b类…

新一代搜索引擎,是 ES 的15倍?

Manticore Search介绍 Manticore Search 是一个使用 C 开发的高性能搜索引擎&#xff0c;创建于 2017 年&#xff0c;其前身是 Sphinx Search 。Manticore Search 充分利用了 Sphinx&#xff0c;显着改进了它的功能&#xff0c;修复了数百个错误&#xff0c;几乎完全重写了代码…

从0开始,来看看怎么去linux排查Java程序故障

一&#xff0c;前提准备 最基本前提&#xff1a;你需要有liunx环境&#xff0c;如果没有请参考其它文献在自己得到local建立一个虚拟机去进行测试。 有了虚拟机之后&#xff0c;你还需要安装jdk和配置环境变量 1. 安装JDK&#xff08;以OpenJDK 17为例&#xff09; 下载JDK…

MFC开发,给对话框添加垂直滚动条并解决鼠标滚动响应的问题

无论在使用QT或者MFC进行界面开发时&#xff0c;都会出现在一个对话框里面存在好多的选项&#xff0c;导致对话框变得非常长或者非常大&#xff0c;就会显现的不美观&#xff0c;在这种情况下通常是添加一个页面的滚动条来解决这个问题&#xff0c;下面我们就来介绍给MFC的对话…

(二)QT——按钮小程序

目录 前言 按钮小程序 1、步骤 2、代码示例 3、多个按钮 ①信号与槽的一对一 ②多对一&#xff08;多个信号连接到同一个槽&#xff09; ③一对多&#xff08;一个信号连接到多个槽&#xff09; 结论 前言 按钮小程序 Qt 按钮程序通常包含 三个核心文件&#xff1a; m…

QT简单实现验证码(字符)

0&#xff09; 运行结果 1&#xff09; 生成随机字符串 Qt主要通过QRandomGenerator类来生成随机数。在此之前的版本中&#xff0c;qrand()函数也常被使用&#xff0c;但从Qt 5.10起&#xff0c;推荐使用更现代化的QRandomGenerator类。 在头文件添加void generateRandomNumb…

受击反馈HitReact、死亡效果Death Dissolve、Floating伤害值Text(末尾附 客户端RPC )

受击反馈HitReact 设置角色受击标签 (GameplayTag基本了解待补充) 角色监听标签并设置移动速度 创建一个受击技能&#xff0c;并应用GE 实现设置角色的受击蒙太奇动画 实现角色受击时播放蒙太奇动画&#xff0c;为了保证通用性&#xff0c;将其设置为一个函数&#xff0c;并…

C++,STL 命名空间:理解 std 的作用、规范与陷阱

文章目录 引言一、为什么需要 std 命名空间&#xff1f;二、std 命名空间的组成三、使用 std 命名空间的正确姿势1. 显式作用域限定2. 谨慎使用 using 声明3. 头文件中禁止 using namespace std 四、常见陷阱与解决方案陷阱 1&#xff1a;与第三方库命名冲突陷阱 2&#xff1a;…

第11章:根据 ShuffleNet V2 迁移学习医学图像分类任务:甲状腺结节检测

目录 1. Shufflenet V2 2. 甲状腺结节检测 2.1 数据集 2.2 训练参数 2.3 训练结果 2.4 可视化网页推理 3. 下载 1. Shufflenet V2 shufflenet v2 论文中提出衡量轻量级网络的性能不能仅仅依靠FLOPs计算量&#xff0c;还应该多方面的考虑&#xff0c;例如MAC(memory acc…

【ArcGIS遇上Python】批量提取多波段影像至单个波段

本案例基于ArcGIS python,将landsat影像的7个波段影像数据,批量提取至单个波段。 相关阅读:【ArcGIS微课1000例】0141:提取多波段影像中的单个波段 文章目录 一、数据准备二、效果比对二、python批处理1. 编写python代码2. 运行代码一、数据准备 实验数据及完整的python位…

HTB:Administrator[WriteUP]

目录 连接至HTB服务器并启动靶机 信息收集 使用rustscan对靶机TCP端口进行开放扫描 将靶机TCP开放端口号提取并保存 使用nmap对靶机TCP开放端口进行脚本、服务扫描 使用nmap对靶机TCP开放端口进行漏洞、系统扫描 使用nmap对靶机常用UDP端口进行开放扫描 使用nmap对靶机…

vscode+WSL2(ubuntu22.04)+pytorch+conda+cuda+cudnn安装系列

最近在家过年闲的没事&#xff0c;于是研究起深度学习开发工具链的配置和安装&#xff0c;之前欲与天公试比高&#xff0c;尝试在win上用vscodecuda11.6vs2019的cl编译器搭建cuda c编程环境&#xff0c;最后惨败&#xff0c;沦为笑柄&#xff0c;痛定思痛&#xff0c;这次直接和…

亚博microros小车-原生ubuntu支持系列:17 gmapping

前置依赖 先看下亚博官网的介绍 Gmapping简介 gmapping只适用于单帧二维激光点数小于1440的点&#xff0c;如果单帧激光点数大于1440&#xff0c;那么就会出【[mapping-4] process has died】 这样的问题。 Gmapping是基于滤波SLAM框架的常用开源SLAM算法。 Gmapping基于RBp…

FreeRTOS从入门到精通 第十六章(任务通知)

参考教程&#xff1a;【正点原子】手把手教你学FreeRTOS实时系统_哔哩哔哩_bilibili 一、任务通知简介 1、概述 &#xff08;1&#xff09;任务通知顾名思义是用来通知任务的&#xff0c;任务控制块中的结构体成员变量ulNotifiedValue就是这个通知值。 &#xff08;2&#…

数据结构选讲 (更新中)

参考 smWCDay7 数据结构选讲2 by yyc 。 可能会补充的&#xff1a; AT_cf17_final_j TreeMST 的 F2 Boruvka算法 目录 AT_cf17_final_j Tree MSTP5280 [ZJOI2019] 线段树 AT_cf17_final_j Tree MST link 题意 给定一棵 n n n 个点的树&#xff0c;点有点权 w i w_i wi​&am…

【01】共识机制

BTF共识 拜占庭将军问题 拜占庭将军问题是一个共识问题 起源 Leslie Lamport在论文《The Byzantine Generals Problem》提出拜占庭将军问题。 核心描述 军中可能有叛徒&#xff0c;却要保证进攻一致&#xff0c;由此引申到计算领域&#xff0c;发展成了一种容错理论。随着…

春晚舞台上的人形机器人:科技与文化的奇妙融合

文章目录 人形机器人Unitree H1的“硬核”实力传统文化与现代科技的创新融合网友热议与文化共鸣未来展望&#xff1a;科技与文化的更多可能结语 2025 年央视春晚的舞台&#xff0c;无疑是全球华人目光聚焦的焦点。就在这个盛大的舞台上&#xff0c;一场名为《秧BOT》的创意融合…

.NET Core缓存

目录 缓存的概念 客户端响应缓存 cache-control 服务器端响应缓存 内存缓存&#xff08;In-memory cache&#xff09; 用法 GetOrCreateAsync 缓存过期时间策略 缓存的过期时间 解决方法&#xff1a; 两种过期时间策略&#xff1a; 绝对过期时间 滑动过期时间 两…

如何从客观角度批判性阅读分析博客

此文仅以个人博客为例&#xff0c;大量阅读朋友反馈给我的交流让我得知他们所理解我的博客所表达的意思并非我所想表达的&#xff0c;差异或大或小&#xff0c;因人而异。 观点与事实 只有从客观角度反复批判性阅读和分析&#xff0c;才能逐渐清晰观点和事实。 观点不等于事实…