算法学习18:动态规划

算法学习18:动态规划


文章目录

  • 算法学习18:动态规划
  • 前言
  • 一、线性DP
    • 1.数字三角形:f[i][j] = max(f[i - 1][j - 1] + a[i][j], f[i - 1][j] + a[i][j]);
    • 2.1最长上升子序列:f[i] = max(f[i], f[j] + 1);
    • 2.2 打印出最长子序列
    • 3.最长公共子序列:
  • 二、区间dp:
    • 1.石子合并:
  • 总结


前言


在这里插入图片描述


提示:以下是本篇文章正文内容:

一、线性DP

1.数字三角形:f[i][j] = max(f[i - 1][j - 1] + a[i][j], f[i - 1][j] + a[i][j]);



在这里插入图片描述



在这里插入图片描述



// 给定一个数字三角形,从顶部出发,在每一个结点可以选择移动至左下方或者是右下方的结点,
// 直到移动到底层,要求找到一条路径,使得路径上的数字的和最大。
// 输入:第一行包含一个整数n,表示数字三角形的层数
// 接下来的n行,每行包含若干整数,其中第i行表示数字三角形第i层包含的整数。#include <iostream>
#include <algorithm>using namespace std;const int N = 510, INF = 1e9;// (相对的)正无穷 int n;
int a[N][N];// 存出数字三角形
int f[N][N];// 状态 int main()
{scanf("%d", &n);for(int i = 1; i <= n; i ++)for(int j = 1; j <= i; j ++)scanf("%d", &a[i][j]);// 初始化// 注意1:对于左右边界,都要多处理一次。(i+1)		for(int i = 0; i <= n; i ++)for(int j = 0; j <= i + 1; j ++)f[i][j] = -INF;// 负无穷 f[1][1] = a[1][1];for(int i = 2; i <= n; i ++)for(int j = 1; j <= i; j ++)f[i][j] = max(f[i - 1][j - 1] + a[i][j], f[i - 1][j] + a[i][j]);int res = -INF;// 遍历最后一层,找到答案 for(int i = 1; i <= n; i ++) res = max(res, f[n][i]);printf("%d\n", res);return 0;} 


在这里插入图片描述



在这里插入图片描述



2.1最长上升子序列:f[i] = max(f[i], f[j] + 1);



在这里插入图片描述



2.2 打印出最长子序列


// 给定一个长度为n的数列,求数值“严格递增”的子序列的长度最长时多少?#include <iostream>
#include <algorithm>using namespace std;const int N = 1010;int n;
int a[N], f[N], g[N];// 数列   状态  存储i是由那个状态转移过来的。 int main()
{scanf("%d", &n);for(int i = 1; i <= n; i ++) scanf("%d", &a[i]);for(int i = 1; i <= n; i ++){f[i] = 1;// 只有a[i]一个数g[i] = 0; for(int j = 1; j < i; j ++)// 保证递增:a[j] 是 a[i] 的前一个数 if(a[j] < a[i])if(f[i] < f[j] + 1){// 更新 f[i] = f[j] + 1;// 记录一下f[i] 是从哪一个状态转移过来的。 g[i] = j;}		}// 找到答案的下标 int k = 1;for(int i = 1; i <= n; i ++)if(f[k] < f[i]) 	k = i;printf("%d\n", f[k]);// 长度 for(int i = 0, len = f[k]; i < len; i ++){printf("%d ", a[k]);// 根据g数组可以知道f[k]是从那个状态转移的 k = g[k];}return 0;
}


在这里插入图片描述



// 给定一个长度为n的数列,求数值“严格递增”的子序列的长度最长时多少?#include <iostream>
#include <algorithm>using namespace std;const int N = 1010;int n;
int a[N], f[N];// 数列   状态 int main()
{scanf("%d", &n);for(int i = 1; i <= n; i ++) scanf("%d", &a[i]);for(int i = 1; i <= n; i ++){f[i] = 1;// 只有a[i]一个数for(int j = 1; j < i; j ++)// 保证递增:a[j] 是 a[i] 的前一个数 if(a[j] < a[i])f[i] = max(f[i], f[j] + 1); }int res = 0;// 便利所有f[i] for(int i = 1; i <= n; i ++)  res = max(res, f[i]);printf("%d", res);return 0;
}


在这里插入图片描述



在这里插入图片描述



3.最长公共子序列:


在这里插入图片描述



在这里插入图片描述



// 给定两个长度分别为n和m的字符串A和B,
// 求即是A的子序列又是B的子序列的字符串的长度最长是多少?
#include <iostream>
#include <algorithm>using namespace std;const int N = 1010;int n, m;
char a[N], b[N];// 2个字符串 
int f[N][N]; int main()
{scanf("%d%d", &n, &m);scanf("%s%s", a + 1, b + 1);// 注意1:从a[1]开始输入字符串// 从1开始遍历 for(int i = 1; i <= n; i ++)for(int j = 1; j <= m; j ++){f[i][j] = max(f[i - 1][j], f[i][j - 1]);if(a[i] == b[j]) f[i][j] = max(f[i][j], f[i - 1][j - 1] + 1);} printf("%d\n", f[n][m]);return 0;
}

在这里插入图片描述



二、区间dp:

1.石子合并:


在这里插入图片描述



在这里插入图片描述



在这里插入图片描述



在这里插入图片描述



// 设有n堆石子排成一排,其编号为1,2,3,......,n
// 用一个整数描述每堆石子的质量,现在要将这n堆石子合并为一堆。// 例子1:有1 3 5 2四堆石子,我们可以先合并1、2堆,代价为4,得到4 5 2,又合并1、2堆,
// 代价为9,得到9 2,在合并得到11,总代价:4+9+11=24
// 例子2:先合并1和2、3和4堆,代价为4+7,得到4 7,再合并,代价为11,总代价:4+7+11=22
#include <iostream>
#include <algorithm>using namespace std;const int N = 310;int n;
int s[N];// 原始数据 
int f[N][N];int main()
{scanf("%d", &n);for(int i = 1; i <= n; i ++) scanf("%d", &s[i]);// 前缀和数组: for(int i = 1; i <= n; i ++) s[i] += s[i - 1];// 按长度从小到大枚举所有状态:从2开始 // 区间长度为1:不需要代价 for(int len = 2; len <= n; len ++)// 枚举起点: for(int i = 1; i + len - 1 <= n; i ++){// 左右端点: int l = i, r = i + len - 1;f[l][r] = 1e8;// 要初始化为一个比较大的数!!!  // 枚举分界点: for(int k = 1; k < r; k ++)f[l][r] = min(f[l][r], f[l][k] + f[k + 1][r] + s[r] - s[l - 1]);}printf("%d\n", f[1][n]);return 0;} 

在这里插入图片描述


总结

提示:这里对文章进行总结:

💕💕💕

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

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

相关文章

Go语言和Java编程语言的主要区别

目录 1.设计理念&#xff1a; 2.语法&#xff1a; 3.性能&#xff1a; 4.并发性&#xff1a; 5.内存管理&#xff1a; 6.标准库&#xff1a; 7.社区和支持&#xff1a; 8.应用领域&#xff1a; Go&#xff08;也称为Golang&#xff09;和Java是两种不同的编程语言&…

免版权素材库:在营销和宣传中的重要性与应用

title: 免版权素材库&#xff1a;在营销和宣传中的重要性与应用 date: 2024/4/5 18:21:43 updated: 2024/4/5 18:21:43 tags: 免版权素材库营销宣传高质量素材节省成本避免侵权创意启发数字营销 免版权素材库在宣传和营销中的重要性不言而喻。在当今数字化时代&#xff0c;图片…

基于Python的微博舆论分析,微博评论情感分析可视化系统,附源码

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;…

刷题之Leetcode34题(超级详细)

34. 在排序数组中查找元素的第一个和最后一个位置 力扣链接(opens new window)https://leetcode.cn/problems/find-first-and-last-position-of-element-in-sorted-array/ 给定一个按照升序排列的整数数组 nums&#xff0c;和一个目标值 target。找出给定目标值在数组中的开始…

蓝桥杯刷题-01-冶炼金属

冶炼金属-暴力求解 小蓝有一个神奇的炉子用于将普通金属 O 冶炼成为一种特殊金属 X。这个炉子有一个称作转换率的属性 V&#xff0c;V 是一个正整数&#xff0c;这意味着消耗 V 个普通金属 O 恰好可以冶炼出一个特殊金属 X&#xff0c;当普通金属 O 的数目不足 V 时&#xff0c…

c# wpf template ItemsPanel 简单试验

1.概要 2.代码 <Window x:Class"WpfApp2.Window9"xmlns"http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x"http://schemas.microsoft.com/winfx/2006/xaml"xmlns:d"http://schemas.microsoft.com/expression/blend/…

MySQL里面慢查询优化指南:从定位到优化

在数据库管理和性能优化中&#xff0c;慢查询是一个常见的问题&#xff0c;它可能导致应用程序响应缓慢甚至崩溃。在本文中&#xff0c;我们将探讨MySQL中慢查询的原因、定位方法和优化策略&#xff0c;帮助您解决慢查询问题&#xff0c;提升数据库性能。 1. 慢查询的原因 慢查…

vue3和vue2 之 provide/inject 用法区别 ---vue3组件间通讯2

一、为什么用他们&#xff1f; provide/inject 主要用于父组件和子孙组件间通讯&#xff0c;不用在父传子&#xff0c;子传孙&#xff0c;孙传重孙等数据传递了&#xff08;解决了Prop 逐级透传问题&#xff09;。简单的父子组件间传值还是使用props、emits或是defineProps、de…

输入nvidia-smi查看的GPU的容量和在docker里输入df -Th查看的shm有联系吗

nvidia-smi命令用于查看GPU的状态和相关信息&#xff0c;而df -Th命令用于查看文件系统的磁盘使用情况&#xff0c;包括共享内存&#xff08;shm&#xff09;的使用情况。 这两个命令提供了不同层面的信息&#xff0c;因此它们之间没有直接的联系。具体来说&#xff1a; nvidi…

call、apply、bind是如何实现改变this的

call, apply, 和 bind 方法都是通过改变函数的执行上下文&#xff08;this&#xff09;来实现改变 this 的。 1. call 方法 call 方法&#xff1a;call 方法通过将函数作为对象的方法调用&#xff0c;并传递一个新的上下文对象作为第一个参数来改变函数的执行上下文。函数内部可…

从无到有开始创建动态顺序表——C语言实现

顺序表的概念 顺序表的底层结构是数组&#xff0c;对数组的封装&#xff0c;实现了常用的增删改查等接口。在物理结构和逻辑结构都是连续的&#xff0c;物理结构是指顺序表在计算机内存的存储方式&#xff0c;逻辑结构是我们思考的形式&#xff0c;顺序表和数组是类似的&#x…

【御控物联】JavaScript JSON结构转换(14):对象To数组——规则属性重组

文章目录 一、JSON结构转换是什么&#xff1f;二、术语解释三、案例之《JSON对象 To JSON数组》四、代码实现五、在线转换工具六、技术资料 一、JSON结构转换是什么&#xff1f; JSON结构转换指的是将一个JSON对象或JSON数组按照一定规则进行重组、筛选、映射或转换&#xff0…

中医肝胆笔记

目录 肝胆的经络足厥阴肝经足少阳胆经 疏肝健脾的药舒肝益脾颗粒&#xff1a;逍遥丸&#xff1a;疏肝颗粒 -> 疏肝理气的力度大-> 肝郁的程度深&#xff0c;逍遥丸没用的是时候用这个加味逍遥丸 -> 清热的力度最大->适用 肝郁火大&#xff0c;舌苔黄丹栀逍遥丸->…

LangChain Demo | 如何调用stackoverflow并结合ReAct回答代码相关问题

背景 楼主决定提升与LLM交互的质量&#xff0c;之前是直接prompt->answer的范式&#xff0c;现在我希望能用上ReAct策略和能够检索StackOverflow&#xff0c;让同一款LLM发挥出更大的作用。 难点 1. 怎样调用StackOverflow step1 pip install stackspi step 2 from la…

使用阿里云试用Elasticsearch学习:1.7 基础入门——索引管理

我们已经看到 Elasticsearch 让开发一个新的应用变得简单&#xff0c;不需要任何预先计划或设置。 不过&#xff0c;要不了多久你就会开始想要优化索引和搜索过程&#xff0c;以便更好地适合您的特定用例。 这些定制几乎围绕着索引和类型的方方面面&#xff0c;在本章&#xff…

基于单片机的有害气体检查系统设计

**单片机设计介绍&#xff0c;基于单片机的有害气体检查系统设计 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于单片机的有害气体检查系统设计旨在实现对环境中各种有害气体的实时监测与报警&#xff0c;保障人员健康和环境…

numpy库read_excek,读取函数

文件说明&#xff1a; 版本&#xff1a;window 10 python3.10 解决的问题&#xff1a;pandas库如何读取excel文件 使用pandas库读取Excel文件可以通过以下步骤实现&#xff1a; 1.首先&#xff0c;确保你已经安装了pandas库。可以使用以下命令在Python环境中安装pandas库&…

如何使用NumPy处理数组翻转与变形

NumPy是Python中一个强大的库&#xff0c;主要用于处理大型多维数组和矩阵的数学运算。处理数组翻转与变形是NumPy的常用功能。 1.对多维数组翻转 n np.random.randint(0,100,size(5,6))n# 执行结果array([[ 9, 48, 20, 85, 19, 93], [ 1, 63, 20, 25, 19, 44], …

用 Wireshark 解码 H.264

H264&#xff0c;你不知道的小技巧-腾讯云开发者社区-腾讯云 这篇文章写的非常好 这里仅做几点补充 init.lua内容&#xff1a; -- Set enable_lua to false to disable Lua support. enable_lua trueif not enable_lua thenreturn end-- If false and Wireshark was start…