算法入门--动态规划(C++)

深入浅出掌握动态规划核心思想,图文并茂+实战代码

什么是动态规划?

动态规划(Dynamic Programming, DP) 是一种高效解决多阶段决策问题的方法。它通过将复杂问题分解为重叠子问题,并存储子问题的解(避免重复计算),最终解决原问题。

动态规划适用场景

  • 重叠子问题:问题可分解为重复出现的子问题

  • 最优子结构:问题的最优解包含子问题的最优解

  • 无后效性:当前状态一旦确定,后续决策不受之前决策影响

一、从斐波那契数列入门(记忆化搜索)

斐波那契数列是理解DP思想的完美起点:F(0)=0, F(1)=1, F(n)=F(n-1)+F(n-2)

1.1 递归解法(低效)

int fib(int n) {if (n <= 1) return n;return fib(n-1) + fib(n-2);
}

问题:存在大量重复计算,时间复杂度O(2^n)

1.2 记忆化搜索(自顶向下)

#include <vector>
using namespace std;int fibMemo(int n, vector<int>& memo) {if (n <= 1) return n;if (memo[n] != -1) return memo[n]; // 已计算过memo[n] = fibMemo(n-1, memo) + fibMemo(n-2, memo);return memo[n];
}int fib(int n) {vector<int> memo(n+1, -1); // 初始化记忆数组return fibMemo(n, memo);
}

优点:时间复杂度降为O(n),空间复杂度O(n)

二、数塔问题(经典DP)

问题描述:从塔顶到塔底,每次只能向下或右下移动,求最大路径和。

        5/ \8   3/ \ / \12  7  16/ \ / \ / \4   10  11  6

2.1 DP解法思路

  1. 状态定义:dp[i][j]表示从第i行第j列出发到底部的最大路径和

  2. 状态转移:dp[i][j] = a[i][j] + max(dp[i+1][j], dp[i+1][j+1])

  3. 边界条件:最底层dp值等于元素本身

2.2 C++实现

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;int maxPathSum(vector<vector<int>>& tower) {int n = tower.size();vector<vector<int>> dp(n, vector<int>(n, 0));// 初始化最后一行for (int j = 0; j < n; ++j) dp[n-1][j] = tower[n-1][j];// 自底向上计算for (int i = n-2; i >= 0; --i) {for (int j = 0; j <= i; ++j) {dp[i][j] = tower[i][j] + max(dp[i+1][j], dp[i+1][j+1]);}}return dp[0][0];
}int main() {vector<vector<int>> tower = {{5},{8, 3},{12, 7, 16},{4, 10, 11, 6}};cout << "最大路径和: " << maxPathSum(tower) << endl; // 输出: 5+8+7+11=31return 0;
}

三、最长上升子序列(LIS)

问题描述:给定数组,找到最长严格递增子序列的长度 示例:[10,9,2,5,3,7,101,18] -> 最长上升子序列[2,5,7,101]长度为4

3.1 DP解法

  1. 状态定义:dp[i]表示以nums[i]结尾的LIS长度

  2. 状态转移:

  3. 结果:max(dp[0...n-1])

3.2 C++实现

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;int lengthOfLIS(vector<int>& nums) {if (nums.empty()) return 0;vector<int> dp(nums.size(), 1);int maxLen = 1;for (int i = 1; i < nums.size(); ++i) {for (int j = 0; j < i; ++j) {if (nums[i] > nums[j]) {dp[i] = max(dp[i], dp[j] + 1);}}maxLen = max(maxLen, dp[i]);}return maxLen;
}int main() {vector<int> nums = {10,9,2,5,3,7,101,18};cout << "最长上升子序列长度: " << lengthOfLIS(nums) << endl; // 输出4return 0;
}

时间复杂度:O(n²) 优化:二分查找可优化至O(nlogn)

四、0-1背包问题(经典DP)

问题描述:给定n件物品(重量w[i], 价值v[i])和容量为C的背包,如何装包使总价值最大?

4.1 DP解法思路

  1. 状态定义:dp[i][j]表示前i件物品放入容量为j的背包的最大价值

  2. 状态转移:

4.2 C++实现

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;int knapsack(int C, vector<int>& w, vector<int>& v) {int n = w.size();vector<vector<int>> dp(n+1, vector<int>(C+1, 0));for (int i = 1; i <= n; ++i) {for (int j = 1; j <= C; ++j) {if (j < w[i-1]) {  // 注意下标偏移dp[i][j] = dp[i-1][j];} else {dp[i][j] = max(dp[i-1][j], v[i-1] + dp[i-1][j - w[i-1]]);}}}return dp[n][C];
}int main() {vector<int> weights = {2, 3, 4, 5};  // 物品重量vector<int> values = {3, 4, 5, 6};   // 物品价值int capacity = 8;                    // 背包容量cout << "最大价值: " << knapsack(capacity, weights, values) << endl; // 输出: 10 (选3+5+6)return 0;
}

五、动态规划解题步骤总结

  1. 定义状态:明确dp数组的含义

  2. 确定转移方程:关键步骤,找出状态间关系

  3. 初始化:设置边界值

  4. 确定计算顺序:自底向上或自顶向下

  5. 输出结果:从最终状态获取答案

  6. 空间优化:滚动数组等技巧(可选)

六、常见DP模型

问题类型

典型问题

状态转移特征

线性DP

最长上升子序列

沿数组顺序转移

区间DP

矩阵链乘法、石子合并

从小区间向大区间转移

背包问题

0-1背包、完全背包

物品选择决策

树形DP

二叉树最大路径和

在树结构上递归转移

状态压缩DP

旅行商问题(TSP)

用二进制表示状态

掌握动态规划需要多练习!建议从LeetCode简单DP题开始,逐步提升难度。

推荐练习:

  • 70. 爬楼梯

  • 53. 最大子数组和

  • 322. 零钱兑换

  • 1143. 最长公共子序列

通过本文的学习,相信你已经掌握了动态规划的基本思想和常见问题的解决方法。继续坚持练习,很快你就能在算法解题中灵活运用DP技巧!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[2025CVPR]GNN-ViTCap:用于病理图像分类与描述模型

论文结构解析​ 本文采用经典学术论文结构: ​引言​:阐述病理图像分析的挑战与现有方法局限性​相关工作​:系统梳理MIL、视觉语言预训练和生物医学语言模型三大领域​方法​:详细阐述GNN-ViTCap四阶段架构​实验​:在BreakHis和PatchGastric数据集验证性能​讨论​:通…

Java SE--图书管理系统模拟实现

一.设计思路首先这个系统可以由俩种用户使用&#xff0c;分别为管理者用户和普通者用户&#xff0c;根据不同的用户有不同的界面&#xff0c;每个界面有不同的功能。二.代码实现创建三个包和一个类book包&#xff1a;包括Book类和Booklist类Book类&#xff1a;package book; pu…

[RPA] 批量数据抓取指定商品名称信息

影刀RPA案例&#xff1a;批量数据抓取指定商品名称信息流程图&#xff1a;操作步骤&#xff1a;涉及的影刀RPA大致指令&#xff1a; 1. 打开影刀商城页面 2. 使用【填写输入框(web)】指令输入用户名和密码&#xff0c;并点击"登录"按钮 3. 切换到"订单管理"…

我的世界Java版1.21.4的Fabric模组开发教程(十四)方块实体

这是适用于Minecraft Java版1.21.4的Fabric模组开发系列教程专栏第十四章——方块实体。想要阅读其他内容&#xff0c;请查看或订阅上面的专栏。 方块实体(Block Entity) 指的是一种用于存储方块额外数据的方法。但这种数据和为了控制方块状态而在自定义方块类中创建的属性不太…

【UE教程/进阶】UE中的指针与引用

目录直接属性引用共享指针 TSharedPtr实现原理共享引用 TSharedRef弱引用指针 TWeakPtrObject弱指针 FWeakPtr实现原理Object软指针 FSoftObjectPtr原理直接属性引用 在c通过UPROPERTY()宏将属性公开&#xff0c;蓝图中属性类型中的Object Reference。 对一个类型及其子类型的…

早期 CNN 的经典模型—卷积神经网络(LeNet)

目录 LeNet 的设计背景与目标 LeNet 的网络结构&#xff08;经典 LeNet-5&#xff09; 局部感受野详解 一、局部感受野和全连接网络的区别 1. 传统全连接网络的问题 2. 局部感受野的解决方案 二、局部感受野的优势 1. 参数大幅减少 2. 提取局部特征 3. 平移不变性 参数…

RabbitMQ 高级特性之延迟队列

1. 简介 在某些场景下&#xff0c;当生产者发送消息后&#xff0c;可能不需要让消费者立即接收到&#xff0c;而是让消息延迟一段时间后再发送给消费者。 2. 实现方式 2.1 TTL 死信队列 给消息设置过期时间后&#xff0c;若消息在这段时间内没有被消费&#xff0c;就会将消…

uniapp app安卓下载文件 图片 doc xls 数据流文件 app安卓本地路径下载保存

//下载图片 downloadToLocal() {plus.android.requestPermissions([android.permission.WRITE_EXTERNAL_STORAGE],(success) > {uni.saveImageToPhotosAlbum({filePath: /static/x.png,//本地地址success: () > {this.$refs.uToast.show({message: "模版下载成功&am…

Context Engineering:从Prompt Engineering到上下文工程的演进

最近在做Deepresearch以及刷到一个不错的文章&#xff1a;context-engineering-guide &#xff0c;这篇文章揭示了提示工程以及上下文过程在智能体应用开源流程中&#xff0c;包括Deepresearch&#xff0c;MCP在内的一些概念&#xff0c;起到了非常重要的作用&#xff01; Cont…

jenkins部署vue前端项目

文章目录前言一、安装nginx二、jenkins构建项目总结前言 前面已经使用jenkins部署了后端springboot项目&#xff0c;现在开始学习jenkins部署前端Vue项目。 一、安装nginx 访问nginx官网&#xff0c;https://nginx.org/en/download.html下载tar包 上传到服务器目录中 然后到…

设计总监年中复盘:用Adobe XD内容识别布局,告别“手动调距”

时至年中&#xff0c;这不仅是检视上半年项目成果的节点&#xff0c;更是优化团队工作流、为下半年挑战储备动能的关键时期。在海外设计界工作的十余年间&#xff0c;我发现&#xff0c;一个高效的设计团队与一个疲于奔命的团队之间&#xff0c;最大的差别往往就在于是否建立了…

Unity 在Rider中通过Lingma插件使用MCP

环境&#xff1a; Unity 2022.3.12f1 JetBrains Rider 2025.1.4 Lingma 2.5.14 Python 3.13.4 下载包 首先在unity package manager 加入unity-mcp包 https://github.com/justinpbarnett/unity-mcp.git 然后下载uv包&#xff08;要先先下载python&#xff09;,网上很多…

pycharm+SSH 深度学习项目 远程后台运行命令

pycharmSSH 深度学习项目 远程后台运行命令碎碎念&#xff0c;都是实验室里那说关机就关机&#xff0c;说重启就重启的台式机逼得。。学吧记录 运行&#xff1a;nohup /root/miniconda3/bin/python -u "run.py" > /root/log/nohup.log 2>&1 &实时查看日…

【Linux | 网络】应用层(HTTP)

目录一、认识URL二、urlencode和urldecode三、HTTP协议格式&#xff08;使用Fiddler抓包&#xff09;3.1 安装并使用Fiddler抓包3.2 HTTP协议格式3.2.1 HTTP请求3.2.1.1 资源URL路径3.2.1.2 请求方法&#xff08;Method&#xff09;3.2.1.3 Location头字段&#xff08;重定向相…

编程实践:单例模式(懒汉模式+饿汉模式)

说明:本专栏文章有两种解锁方案 1:付费订阅,畅享所有文章 2:免费获取,点击下方链接,关注,自动获取免费链接 https://free-img.400040.xyz/4/2025/04/29/6810a50b7ac8b.jpg 主题:C++ 单例模式 什么是单例模式

破局电机制造四大痛点:MES与AI视觉的协同智造实践

万界星空科技电机行业MES系统解决方案是针对电机制造过程中多工序协同难、质量追溯复杂、设备管理要求高等痛点设计的数字化管理系统。一、电机行业的核心痛点1. 多工序协同困难 电机制造涉及绕线、装配、测试等多道工序&#xff0c;工艺衔接复杂&#xff0c;传统人工调度效率…

HTML 初体验

HTML&#xff08;超文本标记语言&#xff09;全称&#xff1a;HyperText Markup Language。超文本是什么&#xff1f;答&#xff1a;超文本就是网页中的链接。标记是什么&#xff1f;答&#xff1a;标记也叫标签&#xff0c;是带尖括号的文本。需求1&#xff1a;将“我爱中国”…

网络层TCP机制

1.确认应答机制由于发送信息的距离可能较远,可能出现后发的信息先到的情况,怎么办?TCP将每个字节的数据都进行了编号,即为序列号如何分辨一个数据包是普通数据还是应答数据呢2.超时重传由于丢包是一个随机的事件,因此在上述tcp传输的过程中,丢包就存在两种情况但是在发送方的角…

【一起来学AI大模型】微调技术:LoRA(Low-Rank Adaptation) 的实战应用

LoRA&#xff08;Low-Rank Adaptation&#xff09; 的实战应用&#xff0c;使用 Hugging Face 的 peft (Parameter-Efficient Fine-Tuning) 库对大型语言模型进行高效微调。LoRA 因其显著降低资源消耗&#xff08;显存和计算&#xff09;同时保持接近全量微调性能的特点&#x…

RedisJSON 内存占用剖析与调优

一、基础内存模型指针包装 所有 JSON 值&#xff08;标量、对象、数组、字符串等&#xff09;至少占用 8 字节&#xff0c;用于存储一个带类型标记的指针。标量与空容器 null、true、false、小整数&#xff08;静态缓存&#xff09;、空字符串、空数组、空对象 均不分配额外内存…