算法--动态规划(线性DP、区间DP)

这里写目录标题

  • tip
    • 数组下标从0开始还是从1开始
  • 线性DP
    • 数学三角形
      • 介绍
      • 算法思想
      • 例题+代码
    • 最长上升子序列
      • 介绍
      • 算法思想
      • 例题+代码
    • 最长公共子序列
      • 介绍
      • 算法思想
      • 例题+代码
    • 编辑距离
      • 介绍
      • 例题+代码
  • 区间DP问题
    • 石子合并
      • 介绍
      • 算法思想
      • 例题+代码

tip

数组下标从0开始还是从1开始

在这里插入图片描述
如果代码中涉及到数组下标为i-1(有时候哪怕不是同一个数组也符合情况,因为是针对同一组数据进行的多个数组设置),那么我们可以使 i 从 1开始,这样,当 i = 1 时,就取到了[0],如果这个位置有特殊情况,那么这样一来我们也不必使用 if ,直接对 f [0]设置一个特殊值即可

注意,“输入”与“使用”是统一的,即如果输入数组时决定了使用 i 从 1 开始,那么到时候使用或者取元素时,一定要记得是从 1 开始,
如果输入决定了是从0开始,那么到时候取元素的时候,记得从0开始

线性DP

数学三角形

介绍

在这里插入图片描述
从顶部出发,可以向左下或者右下移动,最后形成一条路径,找到一条路径使得路径上的数字之和最大

算法思想

在这里插入图片描述
首先我们对三角形的各个数进行编码,从上到下每行从1开始表上行号,之后,东北方向偏45度进行列号的编排,最左边是1列,之后往右依次是2,3,…,这样一来,我们就可以标定某个元素的位置,这也符合三角形位置的数据在数组内存储时的位置

对于动态规划,仍然是状态表示和状态计算,
状态表示:因为每个元素由一个二维数对进行位置表示,所以,状态表示仍然是一个二维的,如上图所示,属性是max
状态计算:由前面的经验可以总结出:状态计算是一个情况的划分,划分的是上一层的情况,即仅研究上一层即可,之后递推原理会帮助代码完成
上一层的状态有两个,一个来自左上,一个来自右上。
两种划分都是曲线救国的思路,即 f [ i - 1 , j - 1 ] + a[ i , j ],另一个是 f [ i - 1 , j ] + a[ i , j ]
其中 a数组存储了所有的数据,a [ i , j ]就是表示[ i , j ]位置的值

例题+代码

在这里插入图片描述
在这里插入图片描述
首先,a[N][N]数组是用来存储所有的数据的
f[N][N]是状态表示

之后,main函数里
首先读入n
之后由于这组数据会涉及到 i-1 ,所以使用数组存数据时,i从1开始到小于等于n,j从1开始到小于等于 i (因为是三角形),读入到a[][]数组中,给[0][x] [x][0]这些位置空出来,方便一些特殊情况时拿来使用(包括a数组自己使用或者其他数组使用,总之这个位置要留出来)
之后初始化状态表示,因为状态表示数组涉及到某一点的上边两个角,所以,要处理好边界问题,我们的 i 从 0开始到等于n,j 从 0开始到等于 i + 1 (这样做可以在原来的数据基础上,多加一个0行和0列被初始化,这样的话,当我们遇到如下图所示情况时,就不会出错了),且f[i][j] = - INF,将所有的位置初始化为负无穷,这样的话,最终原数据的外面的位置下标,f[][]的值是负无穷,这样做是为了当我们有路径来到了原数据的外面,那最后的值一定是负无穷,这样我们就可以筛选出来哪些路径是经过了原数据的外面的位置得到的,这些数据可以排除
之后初始化f[1][1] = a[1][1],因为第一个还没有经过任何其他的点,可以初始化为1
之后for循环,i 从 2 开始到等于n(之所以从2开始,是因为1就一个节点,已经被初始化了)
二层循环 ,j 从 1 开始到等于i (仍然是因为输入时是三角形输入)
f[ i ][ j ] = max(f[ i - 1 ][j - 1] + a[ i ][ j ] , f[ i - 1 ][ j ] + a[ i ][ j ])
到此为止,我们就已经可以拿到所有 i j 点的状态了,我们由于是想求最大路径,所以对叶子结点的状态进行求max即可,这里由于之前所有的f都被初始化为了负无穷,所以这样答案也初始化为负无穷去进行比较,而就算有路径经过了负无穷,那最后的值是-INF+一些正数,肯定比-INF要大一点点,所以,res用-INF存储,最后参与到max中
在这里插入图片描述

最长上升子序列

介绍

在这里插入图片描述
在这里插入图片描述
例如这样一个序列,最长子序列就是 1 2 5 6,长度最长是4

算法思想

在这里插入图片描述
状态表示可以使用一维的,即所有以“第” i 个数结尾的上升子序列,属性是max
状态计算:
划分依据:上一层是第几个数
如果上一层没有数,那么说明只有“第i个数”这一个数
如果上一层的数是第 j 个数,那么可以递推为 f [ j ] + 1(这个j是有条件的,要满足aj < ai)
最后对所有的f[ j ] + 1 (j从0开始到 i - 1)取max就是答案
在这里插入图片描述

例题+代码

在这里插入图片描述
在这里插入图片描述
a数组表示将数据进行存储,f数组表示状态表示
在mian函数中
首先输入n
i 从 1 到等于n ,输入a[ i ]
之后对for从1到等于n
f[ i ] = 1 ,(所有的 f 最少是1,所以先初始化为1)
之后,对于for循环,j 从 1 开始到小于 i (因为状态计算时,是以" "前一个数"是第几个数 " 进行划分的,所以j从1到小于i,表示遍历所有划分的情况,为后续取max做准备,这样就可以找到最大的子序列对应的是哪个 j ,因为每个j对应的f[j] 是不同的)
同时判断 if (a[j] < a[i])才是合法的(因为是上升子序列,要求单调)
之后 f[i] = max(f[i] , f[j]+1)(对所有的划分进行max)
在这里插入图片描述

最后,把所有的 f[i] 进行取max,即,将所有的情况取最大值,就是最大子序列

补充:
在这里插入图片描述
在“动态规划(2)”中00:54时,介绍了如何将最长子序列保存下来

最长公共子序列

介绍

在这里插入图片描述

算法思想

在这里插入图片描述
首先,对于状态表示,集合方面,表示所有在第一个序列的前i个字母中出现,且在第二个序列的前 j 个字母中出现的公共子序列,属性是max

之后对于状态计算
可以对公共子序列是否包含a[i] b[j]进行划分,0表示不包含,1表示包含,如上图
之后对于00:
f [ i - 1 , j - 1]
对于11:
f [ i - 1 , j - 1]
对于上面这两个,毫无疑问,确实是这样表示的,
但是对于01和10,我们下面这种表示,是错误的,拿01举例来说,我们想要的是不包含ai但是必须要有bj,而f [i - 1 , j ]表示不包含ai,但是可以有bj也可以没有,所以他是错的,但是f [i - 1 , j ]包含了01这种情况,由于我们是求最大值,而且f的属性也是max,所以,既然f [i - 1 , j ]包含了01的情况,那么就可以拿来进行对比,因为如果对于f [i - 1 , j ]来说,如果01是其所有情况的最大值,那么就符合我们的要求,如果不是,那么我们也没必要再关注01了,而是其他某种情况,所以可以直接用f [i - 1 , j ]。10同理
对于01:
f [i - 1 , j ]
对于10:
f [ i , j - 1 ]

最后补充:
在这里插入图片描述
我们在写代码时,不用写00的情况,因为这种情况也被包含在了f [i - 1 , j ]和f [i , j - 1]两种情况里,所以我们在写代码时只需要考虑后面三种情况即可

例题+代码

在这里插入图片描述
在这里插入图片描述
n和m表示a和b字符串的长度
char a b数组用来存储字符串
f数组是状态表示

之后在main里:
首先输入n 和 m
之后输入字符串,使用%s,以及数组名(这里是数组名+1,可以从数组的第二个位置,也就是下标为1的位置开始将整个字符串都读入,因为代码中出现了i-1)
之后for i 从1~=n
二重循环 j 从1~=m(也就是将所有的 f 进行了遍历)
f[i][j] = max (f[i-1][j] , f[i][j-1]),先将这两个进行取max
之后对11这种情况进行判断
只有当a[i] == b[j]时,才会有第三种情况,所以只有该条件成立,将第三种情况放入max中
最后输出 f[n][m]

注意,因为我们是存储字符串,a b数组一定要用char数组,而不能使用int数组,首先这是一个规范问题,其次:
在这里插入图片描述
因为int型有四个字节,而字符只占一个字节,所以,从a[ 1 ]开始存入四个字符的话,就会如上图所示效果,四个字符才占满一个int,从而会导致程序错误

编辑距离

介绍

例题+代码

在这里插入图片描述

区间DP问题

石子合并

介绍

在这里插入图片描述
不同的合并顺序,所耗费的总代价是不同的,这是因为我们每次合并的之后产生的代价都会被计算在内,所以,虽然最后的总质量是不变的,但是前期每一个步骤花费的代价都被计算在内,所以,关键就在于前期合并时的代价要尽可能的小

算法思想

在这里插入图片描述
状态表示如上图所示,属性为min
状态计算:
划分的依据是合并成 f[i , j ]的过程中,最后一次合并时,左右两边的大堆是由多少个小堆合并而来的。
我们以左边大堆的基础堆数为依据进行图示,如上图,从1、2、3、…k-1,k表示基础石碓的总堆数:j-i+1
在这里插入图片描述
接上面状态计算,我们这里由数量转换到区间,注意,这时这里的k表示区间的分界点,而不是总堆数,代码中或者理解上换个字母也可以,
从 i 堆到 j 堆,我们看这个过程中最后一堆的合成,我们假设是区间【i,k】和区间【k+1,j】,分别表示左边大堆和右边大堆,然后,因为根据题意,我们最后还有加上目前的总研究范围内的堆数之和,因为最后总会加上总堆数,而目前我们的总研究范围是i到j,所以我们不能确定 i 和 j 是什么,所以使用前缀和进行计算,即s[ j ] - s[ i-1 ],表示 i 到 j 的总堆数。
因为总堆数是一个定数,所以划分上一层时,可以将总堆数排除计算,最后再加回来即可,所以就是用f[i , k] + f[k + 1 , j]进行划分,最后加上总堆数s[ j ] - s[ i-1 ]
而我们的f[i , j]=min{f[i , k] + f[k + 1 , j]+s[ j ] - s[ i-1 ]},k从i遍历到j-1(之所以遍历到j-1,是因为右边堆最少有一个基础堆,所以不能是j。表示左边堆的数量从1到k-1)
这里之所以k从 i 到 j-1,是因为当k=i时,区间划分为【i,i】【i+1,j】,这是左边只有一堆的情况,当k=j-1时,区间为【i,j-1】【j,j】,这是右边只有一堆的情况,两边的端点情况都被包括了。是合理的
时间复杂度是o(n的三次方)

例题+代码

在这里插入图片描述

在这里插入图片描述
注意,因为区间dp的递归中,没有明显的归的代码,也就是一些基础数据没有被计算,所以我们在做区间划分递归时,要用一个手段,来保证每次状态的计算所用到的状态都已经被计算好了,这个手段就是根据区间长度进行循环,先算区间长度小的状态

数据分析:s数组用来计算前缀和,f数组是状态表示

在main函数中:
首先输入n
之后,i从1~=n输入各个堆的重量到s数组中,之后i从1到等于n,进行前缀和的求和:s[ i ] += s[ i -1]
之后开始长度循环,长度从1开始到小于等于n,但是由于长度为1时,总共只有一堆石子,没有仍和代价的耗费,而f数组定义在全局位置,所以自动初始化为0,所以不用管len=1的情况,len直接从2开始到小于等于n
之后二重循环,对起点进行循环,因为给定一个长度之后,不同的起点对应不同的石子,所以起点 i 从1到 i + len -1 <= n(即由起点确定的右端点不能超出石子的个数n),i++
之后有了长度和起点,就可以确定左右端点了,l = i ,r = i + len - 1
之后初始化f[ l ][ r ] = 1e8,要对f数组初始化为一个很大的数,因为我们最终要取min,所以初始化时尽可能大
最后for循环k从l 到 小于r
f[ l ][ r ] = min (…)

最后输出f[ 1 ][ n ]即可

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

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

相关文章

Topaz Video AI:一键提升视频品质,智能重塑影像魅力 mac/win版

Topaz Video AI是一款革命性的视频智能处理软件&#xff0c;它利用先进的机器学习和人工智能技术&#xff0c;为视频创作者提供了前所未有的视频增强和修复功能。无论您是专业视频编辑师、摄影师&#xff0c;还是热爱视频创作的爱好者&#xff0c;Topaz Video AI都能帮助您轻松…

返回JSON对象

在目前的Java项目中&#xff0c;我们最经常使用的便是JSON&#xff0c;不是传递JSON对象&#xff0c;就是返回JSON对象&#xff0c;甚至还把多个参数封装成JSON对象来进行传递&#xff0c;以便简化代码等&#xff01; 但是&#xff0c;该如何操作代码才能正确的返回一个或者多…

excel导出标准化

虽然标题叫标准化&#xff0c;只不过是我自己的习惯&#xff0c;当一件事情变得流程标准化之后&#xff0c;开发程序就会飞快&#xff0c;开发评估工作总是 搞个1~2天&#xff0c;实则前端后端一起开发&#xff0c;1个小时就可以搞定。 1 前端 const exportXls async () >…

每日一题——LeetCode1556.千位分隔符

方法一 个人方法&#xff1a; 把n转为字符串&#xff0c;逆序遍历n&#xff0c;把n的每个元素加入res&#xff0c;每三次加入.&#xff0c;最后将res翻转再转为字符串即为符合题目要求的结果 var thousandSeparator function(n) {nlet res[],lenn.length-1for(let ilen;i>…

tomcat 搭建博客 及破解数据库密码

一 tomcat 搭建博客 &#xff08;一&#xff09;博客安装包 1&#xff0c; 把博客war包 放到 webapps 文件夹下 2&#xff0c;会自动解压 3&#xff0c;做个软连接 方便后续操作 可以注意到 因为war包 是又tomcat 自己解压的 所以属主数组还是 tomcat &#xff08…

大模型生成,Open API调用

大模型是怎么生成结果的 通俗原理 其实&#xff0c;它只是根据上文&#xff0c;猜下一个词&#xff08;的概率&#xff09;…… OpenAI 的接口名就叫【completion】&#xff0c;也证明了其只会【生成】的本质。 下面用程序演示【生成下一个字】。你可以自己修改 prompt 试试…

LaMa Image Inpainting 图像修复 Onnx Demo

目录 介绍 效果 模型信息 项目 代码 下载 LaMa Image Inpainting 图像修复 Onnx Demo 介绍 gihub地址&#xff1a;https://github.com/advimman/lama &#x1f999; LaMa Image Inpainting, Resolution-robust Large Mask Inpainting with Fourier Convolutions, WAC…

电脑休眠之后唤不醒

现象&#xff1a;午休时间电脑休眠了&#xff0c;醒来之后发现在密码输入界面&#xff0c;但鼠标键盘没反应。按重启键或电源机重新开机&#xff0c;结果开不了机。 原因&#xff1a;1、内存条脏了&#xff0c;导致内存条读取失败 2、休眠的时候硬盘休眠了&#xff0c;导致按…

7、Redis-事务、持久化、内存淘汰机制和过期key处理

目录 一、事务 二、持久化 三、内存淘汰机制 四、过期key处理 一、事务 Redis的事务本质上就是一个批量执行命令的操作。分为三个步骤&#xff1a; 开始事务&#xff1a;multi命令入队&#xff1a;正常输入命令即可执行事务&#xff08;依次执行命令&#xff09;&#xf…

linux_day04

大纲&#xff1a;命令&#xff0c;vim&#xff0c;gcc&#xff0c;编译工具&#xff0c;生成代码&#xff0c;调试&#xff0c;库makefile&#xff0c;系统编程 文件系统&#xff1a;文件属性&#xff0c;文件内容&#xff0c;万物皆文件&#xff08;不在内存中的是文件&#…

为什么说?上位机开发有广泛的前景

上位机开发展现了广泛的前景&#xff0c;主要有以下几个方面的原因&#xff1a; 广泛应用的C#语言&#xff1a; C#在软件开发领域得到了广泛应用&#xff0c;拥有丰富的库、工具和社区支持&#xff0c;使得学习和使用C#进行上位机开发更加便捷。与Windows密切相关&#xff1a; …

LeetCode -- 79.单词搜索

1. 问题描述 给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 单词必须按照字母顺序&#xff0c;通过相邻的单元格内的字母构成&#xff0c;其中“相邻”单元格是那些水…

用 Pytest+Allure 生成漂亮的 HTML 图形化测试报告

本篇文章将介绍如何使用开源的测试报告生成框架 Allure 生成规范、格式统一、美观的测试报告。 通过这篇文章的介绍&#xff0c;你将能够&#xff1a; 将 Allure 与 Pytest 测试框架相结合&#xff1b;如何定制化测试报告内容执行测试之后&#xff0c;生成 Allure 格式的测试报…

10.广域网技术

1. PPP实验点这里&#xff08;拓扑代码&#xff09; 2. PPPoE配置实验点这里&#xff08;拓扑代码&#xff09; 目录 一、广域网二、PPP协议三、PPP链路建立过程1-LCP&#xff08;链路协商&#xff09;四、PPP链路建立过程2-PAP/CHAP&#xff08;认证协商&#xff0c;可选&…

微服务间通信重构与服务治理笔记

父工程 依赖版本管理,但实际不引入依赖 pom.xml <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation&…

详解字符串函数<string.h>(上)

1. strlen函数的使用和模拟实现 size_t strlen(const char* str); 1.1 函数功能以及用法 字符串长度 strlen函数的功能是计算字符串的长度。在使用时&#xff0c;要求用户传入需要计算长度的字符串的起始位置&#xff0c;并返回字符串的长度。 #include <stdio.h> #…

【两万字面试系列】三年前的面试题。Service里面的线程安全问题

前言 三年前&#xff0c;大概是21年&#xff0c;那会刚学完java&#xff0c;然后去面试&#xff0c;被打的一塌糊涂&#xff0c;今天来盘一盘之前的面试&#xff0c;到底是怎样的问题整住了。然后发现了去年整的线程安全东西&#xff0c;也贴到文章后面了。那个贴的还不太准&a…

2024腾讯云服务器优惠价格表又降价了,给同行干emo了

腾讯云优惠活动2024新春采购节活动上线&#xff0c;云服务器价格已经出来了&#xff0c;云服务器61元一年起&#xff0c;配置和价格基本上和上个月没什么变化&#xff0c;但是新增了8888元代金券和会员续费优惠&#xff0c;腾讯云百科txybk.com整理腾讯云最新优惠活动云服务器配…

探索数据结构:解锁计算世界的密码

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;数据结构与算法 贝蒂的主页&#xff1a;Betty‘s blog 前言 随着应用程序变得越来越复杂和数据越来越丰富&#xff0c;几百万、…

每日五道java面试题之spring篇(九)

目录&#xff1a; 第一题. 说一下Spring的事务传播行为第二题. 说一下 spring 的事务隔离&#xff1f;第三题. Spring AOP and AspectJ AOP 有什么区别&#xff1f;AOP 有哪些实现方式&#xff1f;第四题. JDK动态代理和CGLIB动态代理的区别第五题. 解释一下Spring AOP里面的几…