动态规划(一)

news2025/1/16 16:54:55

目录

(一)递归到动规的一般转化方法

(二)动规解题的一般思路 

1. 将原问题分解为子问题

2. 确定状态

3. 确定一些初始状态(边界状态)的值

4. 确定状态转移方程

(三)能用动规解决的问题的特点

1.最优子结构

2.无后效性

(四)动归的常用两种形式 

1)递归型

2)递推型

(五)例题

数字三角形

题目

解题思路

题目解答

运行该程序会超时,为什么呢?

递归改递推

空间优化

 最长上升子序列

题目

解题思路

1.找子问题

2. 确定状态

3. 找出状态转移方程

题目解答

 公共子序列 

题目

解题思路

题目解答


(一)递归到动规的一般转化方法


        递归函数有n个参数,就定义一个n维的数组,数组的下标是递归函数参数的取值范围,数组元素的值是递归函数的返回值,这样就可以从边界值开始,逐步填充数组,相当于计算递归函数值的逆过程。

(二)动规解题的一般思路 


1. 将原问题分解为子问题

        把原问题分解为若干个子问题,子问题和原问题形式相同或类似,只不过规模变小了。子问题都解决,原问题即解决(数字三角形例)。
子问题的解一旦求出就会被保存,所以每个子问题只需求解一次。

2. 确定状态

        在用动态规划解题时,我们往往将和子问题相关的各个变量的一组取值,称之为一个“状态”。一个“状态”对应于一个或多个子问题,所谓某个“状态”下的“值”,就是这个“状态”所对应的子问题的解。
        用动态规划解题,经常碰到的情况是,K个整型变量能构成一个状态(如数字三角形中的行号和列号这两个变量构成“状态”)。如果这K个整型变量的取值范围分别是N1, N2, ……Nk,那么,我们就可以用一个K维的数组array[N1] [N2]……[Nk]来存储各个状态的“值”。这个“值”未必就是一个整数或浮点数,可能是需要一个结构才能表示的,那么array就可以是一个结构数组。一个 “状态”下的“值”通常会是一个或多个子问题的解。 

3. 确定一些初始状态(边界状态)的值

        以“数字三角形”为例,初始状态就是底边数字,值就是底边数字值。

4. 确定状态转移方程

        定义出什么是“状态”,以及在该 “状态”下的“值”后,就要找出不同的状态之间如何迁移――即如何从一个或多个“值”已知的 “状态”,求出另一个“状态”的“值”(“人人为我”递推型)。状态的迁移可以用递推公式表示,此递推公式也可被称作“状态转移方程”。

(三)能用动规解决的问题的特点


1.最优子结构

        问题具有最优子结构性质。如果问题的最优解所包含的子问题的解也是最优的,我们就称该问题具有最优子结构性质。

2.无后效性

        无后效性。当前的若干个状态值一旦确定,则此后过程的演变就只和这若干个状态的值有关,和之前是采取哪种手段或经过哪条路径演变到当前的这若干个状态,没有关系。

(四)动归的常用两种形式 


1)递归型

        优点:直观,容易编写
        缺点:可能会因递归层数太深导致爆栈,函数调用带来额外时间开销。无法使用滚动数组节省空间。总体来说,比递推型慢。

2)递推型

        效率高,有可能使用滚动数组节省空间

(五)例题

数字三角形

题目

        图给出了一个数字三角形。从三角形的顶部到底部有很多条不同的路径。对于每条路径,把路径上面的数加起来可以得到一个和,你的任务就是找到最大的和。
        注意:路径上的每一步只能从一个数走到下一层上和它最近的左边的那个数或者右边的那个数。

输入

        输入的是一行是一个整数N (1 < N <= 100),给出三角形的行数。下面的N行给出数字三角形。数字三角形上的数的范围都在0和100之间。

5
7
3 8
8 1 0 
2 7 4 4
4 5 2 6 5

输出

        输出最大的和。

30

解题思路

用二维数组存放数字三角形。
D( r, j)   : 第r行第 j 个数字(r,j从1开始算)
MaxSum(r, j) :   从D(r,j)到底边的各条路径中,
                          最佳路径的数字之和。
问题:求 MaxSum(1,1)
典型的递归问题。
D(r, j)出发,下一步只能走D(r+1,j)或者D(r+1, j+1)。故对于N行的三角形:
if ( r == N)
        MaxSum(r,j) = D(r,j)
else
        MaxSum( r, j) = Max{ MaxSum(r+1,j), MaxSum(r+1,j+1) }+ D(r,j)

题目解答

#include <iostream> 
#include <algorithm>
#define MAX 101 
using namespace std;
int D[MAX][MAX]; 
int n; 
int MaxSum(int i, int j){
    if(i==n)
        return D[i][j]; 
    int x = MaxSum(i+1,j); 
    int y = MaxSum(i+1,j+1); 
    return max(x,y)+D[i][j];
}
int main(){
    int i,j;
    cin >> n;
    for(i=1;i<=n;i++)
        for(j=1;j<=i;j++)
            cin >> D[i][j];
    cout << MaxSum(1,1) << endl;
}

运行该程序会超时,为什么呢?

        原因是重复计算,如果采用递规的方法,深度遍历每条路径,存在大量重复计算。则时间复杂度为 2n,对于 n = 100 行,肯定超时。

改进

        如果每算出一个MaxSum(r,j)就保存起来,下次用到其值的时候直接取用,则可免去重复计算。那么可以用O(n2)时间完成计算。因为三角形的数字总数是 n(n+1)/2

#include <iostream> 
#include <algorithm> 
using namespace std;
#define MAX 101 
int D[MAX][MAX];    
int n; 
int maxSum[MAX][MAX];
int MaxSum(int i, int j)
{
    if( maxSum[i][j] != -1 )
        return maxSum[i][j]; 
    if(i==n)  
        maxSum[i][j] = D[i][j]; 
    else {
        int x = MaxSum(i+1,j); 
        int y = MaxSum(i+1,j+1); 
        maxSum[i][j] = max(x,y)+ D[i][j];
    }
    return maxSum[i][j];
    
int main(){
    int i,j; 
    cin >> n; 
    for(i=1;i<=n;i++)
        for(j=1;j<=i;j++) {
                cin >> D[i][j]; maxSum[i][j] = -1;
        }
    cout << MaxSum(1,1) << endl;
}

递归改递推

#include <iostream> 
#include <algorithm>
using namespace std;
#define MAX 101 
int D[MAX][MAX];   
int n; 
int maxSum[MAX][MAX]; 

int main(){ 
    int i,j; 
    cin >> n; 
    for(i=1;i<=n;i++) 
        for(j=1;j<=i;j++) 
            cin >> D[i][j];
    for( int i = 1;i <= n; ++ i )
        maxSum[n][i] = D[n][i];
    for( int i = n-1; i>= 1;  --i ) 
        for( int j = 1; j <= i; ++j ) 
            maxSum[i][j] =max(maxSum[i+1][j],maxSum[i+1][j+1]) + D[i][j];
    cout << maxSum[1][1] << endl; 
}

空间优化

        没必要用二维maxSum数组存储每一个MaxSum(r,j),只要从底层一行行向上递推,那么只要一维数组maxSum[100]即可,即只要存储一行的MaxSum值就可以。

        进一步考虑,连maxSum数组都可以不要,直接用D的第n行替代maxSum即可。

        节省空间,时间复杂度不变

#include <iostream> 
#include <algorithm> 
using namespace std;
#define MAX 101 
int D[MAX][MAX]; 
int n; 
int * maxSum; 

int main(){ 
    int i,j; 
    cin >> n; 
    for(i=1;i<=n;i++) 
        for(j=1;j<=i;j++) 
        cin >> D[i][j]; 
    maxSum = D[n]; //maxSum指向第n行
    for( int i = n-1; i>= 1;  --i ) 
        for( int j = 1; j <= i; ++j ) 
            maxSum[j] = max(maxSum[j],maxSum[j+1]) + D[i][j]; 
    cout << maxSum[1] << endl; 
}

 最长上升子序列

题目

        一个数的序列bi,当b1 < b2 < ... < bS的时候,我们称这个序列是上升的。对于给定的一个序列(a1, a2, ..., aN),我们可以得到一些上升的子序列(ai1, ai2, ..., aiK),这里1 <= i1 < i2 < ... < iK <= N。比如,对于序列(1, 7, 3, 5, 9, 4, 8),有它的一些上升子序列,如(1, 7), (3, 4, 8)等等。这些子序列中最长的长度是4,比如子序列(1, 3, 5, 8).
        你的任务,就是对于给定的序列,求出最长上升子序列的长度。

输入 

        输入的第一行是序列的长度N (1 <= N <= 1000)。第二行给出序列中的N个整数,这些整数的取值范围都在0到10000。 

7
1 7 3 5 9 4 8

输出 

        最长上升子序列的长度。 

4

解题思路

1.找子问题

        “求序列的前n个元素的最长上升子序列的长度”是个子问题,但这样分解子问题,不具有“无后效性”
        假设F(n) = x,但可能有多个序列满足F(n) = x。有的序列的最后一个元素比 an+1小,则加上an+1就能形成更长上升子序列;有的序列最后一个元素不比an+1小……以后的事情受如何达到状态n的影响,不符合“无后效性”
        “求以ak(k=1, 2, 3…N)为终点的最长上升子序列的长度”
        一个上升子序列中最右边的那个数,称为该子序列的 “终点”。
        虽然这个子问题和原问题形式上并不完全一样,但是只要这N个子问题都解决了,那么这N个子问题的解中,最大的那个就是整个问题的解。 

2. 确定状态

        子问题只和一个变量-- 数字的位置相关。因此序列中数的位置k 就是“状态”,而状态 k 对应的“值”,就是以ak做为 “终点”的最长上升子序列的长度。
        状态一共有N个。

3. 找出状态转移方程

        maxLen (k)表示以ak做为“终点”的最长上升子序列的长度那么:
        初始状态:maxLen (1) = 1
        maxLen (k) = max { maxLen (i):1<=i < k 且 ai < ak且 k≠1 } + 1
                若找不到这样的i,则maxLen(k) = 1
        maxLen(k)的值,就是在ak左边,“终点”数值小于ak ,且长度最大的那个上升子序列的长度再加1。因为ak左边任何“终点”小于ak的子序列,加上ak后就能形成一个更长的上升子序列。

题目解答

#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
const int MAXN = 1010;
int a[MAXN];
int maxLen[MAXN];

int main(){
    int N;
    cin >> N;
    for(int i = 1;i <= N;++i){
        cin >>a[i];
        maxLen[i] = 1;
    }
    for(int i = 2;i <= N;++i){
        //每次求以第i个数为终点的最长上升子序列的长度
        for( int j = 1; j < i; ++j) 
            //察看以第j个数为终点的最长上升子序列
            if( a[i] > a[j] )
                maxLen[i] = max(maxLen[i],maxLen[j]+1); 
    }
    cout << * max_element(maxLen+1,maxLen + N + 1 );
    return 0;
}} //时间复杂度O(N2)

 公共子序列 

题目

        我们称序列Z = < z1, z2, ..., zk >是序列X = < x1, x2, ..., xm >的子序列当且仅当存在 严格上升 的序列< i1, i2, ..., ik >,使得对j = 1, 2, ... ,k, 有xij = zj。比如Z = < a, b, f, c > 是X = < a, b, c, f, b, c >的子序列。
        现在给出两个序列X和Y,你的任务是找到X和Y的最大公共子序列,也就是说要找到一个最长的序列Z,使得Z既是X的子序列也是Y的子序列。

输入

        输入包括多组测试数据。每组数据包括一行,给出两个长度不超过200的字符串,表示两个序列。两个字符串之间由若干个空格隔开。

abcfbc         abfcab
programming    contest 
abcd           mnp

输出

        对每组输入数据,输出一行,给出两个序列的最大公共子序列的长度。

4
2
0

解题思路

输入两个串s1,s2,设MaxLen(i,j)表示:  
s1的左边i个字符形成的子串,与s2左边的j个字符形成的子串的最长公共子序列的长度(i,j从0开始算)
MaxLen(i,j) 就是本题的“状态”
假定 len1 = strlen(s1),len2 = strlen(s2)
那么题目就是要求 MaxLen(len1,len2)
显然:
MaxLen(n,0)  = 0  ( n= 0…len1)
MaxLen(0,n)  = 0  ( n=0…len2)
递推公式:
if ( s1[i-1] == s2[j-1] ) //s1的最左边字符是s1[0]
        MaxLen(i,j) = MaxLen(i-1,j-1) + 1;
else
        MaxLen(i,j) = Max(MaxLen(i,j-1),MaxLen(i-1,j) );
时间复杂度O(mn) m,n是两个字串长度

S1[i-1]!= s2[j-1]时,MaxLen(S1,S2)不会比MaxLen(S1,S2j-1) 和MaxLen(S1i-1,S2)两者之中任何一个小,也不会比两者都大。 

题目解答

#include <iostream>
#include <cstring>
using namespace std;
char sz1[1000];
char sz2[1000];
int maxLen[1000][1000];

int main(){
    while( cin >> sz1 >> sz2 ){
        int length1 = strlen(sz1);
        int length2 = strlen(sz2);
        int nTmp;
        int i,j;
        for(i = 0;i <= length1;i++)
            maxLen[i][0] = 0;
        for(j = 0;j <= length2;j++)
            maxLen[0][j] = 0;
        for(i = 1;i <= length1;i++){
            for(j = 1; j <= length2;j++){
                if(sz1[i-1] == sz2[j-1])
                    maxLen[i][j] = maxLen[i-1][j-1] + 1;
                else
                   maxLen[i][j] = max(maxLen[i][j-1],maxLen[i-1][j]);                 
            }
        }
        cout <<  maxLen[length1][length2] << endl;
    }
    return 0;
}

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

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

相关文章

【Android Studio】Webview 内核升级得三种方法

【Android Studio】Webview 内核升级得三种方法 前言X5 腾讯组件crosswalk开源项目webview升级加载的内核&#xff08;完美解决&#xff09;总结 前言 在APP 中进行网页加载&#xff0c;一般采用原生自带的Webview 组件&#xff0c;但在需要加载高版本网页的时候&#xff0c;有…

工业三防平板助力MES系统打造工厂移动式生产管理

随着工业4.0时代的到来&#xff0c;智能制造、数字化车间等概念层出不穷&#xff0c;生产过程的可视化管理也成为了企业提升效率、优化生产的关键。而工业三防平板&#xff0c;凭借其坚固耐用、功能强大、便携易用等特性&#xff0c;成为了实现生产过程可视化管理的重要利器&am…

SQL注入实例(sqli-labs/less-21)

与第20关无异&#xff0c;只多了一步base64加密 0、初始页面 1、确定闭合字符 2、爆库名 3、爆表名 4、爆列名 5、查询最终目标

POS刷卡开发源码之语音播报-CyberWinApp-SAAS 本地化及未来之窗行业应用跨平台架构

一、终端语音提醒的好处 1. 增强信息传递的有效性&#xff1a;在人们忙碌或者注意力分散时&#xff0c;语音提醒能够直接穿透噪音和干扰&#xff0c;确保重要信息被准确接收。 2. 提高操作的便捷性&#xff1a;用户无需停下手中的工作去查看屏幕或阅读文字&#xff0c;直接通过…

算法——动态规划:0/1 背包问题

文章目录 一、问题描述二、解决方案1. DP 状态的设计2. 状态转移方程3. 算法复杂度4. 举例5. 实现6. 滚动数组6.1 两行实现6.2 单行实现6.3 优缺点 三、总结 一、问题描述 问题的抽象&#xff1a;给定 n n n 种物品和一个背包&#xff0c;第 i i i 个物品的体积为 c i c_i …

NET8中WebAPI使用JWT入门教程

目录 1、JWT2、具体实现3、代码下载 1、JWT 现在在各类API的开发中&#xff0c;token已经是必备了。例如&#xff1a;微信公众号开发中&#xff0c;第一个方法就是获取token。JWT具体的定义及组成部分大家可以到网上找找&#xff0c;这儿给一个简单的描述&#xff1a;JWT 令牌…

yaml语法+yaml配置文件

yaml语法 k:(空格)v > 表示一对键值对空格必须有 yaml拥有严格的空格缩进格式控制&#xff0c;以空格的缩进来控制层级关系&#xff1b;只要是左对齐的一列数据&#xff0c;都是同一个层级的 spring:thymeleaf:cache: true# 检查模板是否存在&#xff0c;然后再呈现check…

通义灵码-阿里云推出的AI智能编码助手

通义灵码体验地址 标题通义灵码是什么&#xff1f; 通义灵码是由阿里巴巴推出的基于通义大模型的智能编码辅助工具&#xff0c;提供行级/函数级实时续写、自然语言生成代码、单元测试生成、代码注释生成、代码解释、研发智能问答、异常报错排查等能力&#xff0c;并针对阿里云…

Bug 解决 | 前端无法正确请求后端接口并得到响应?

目录 1、配置问题 2、代码问题 3、网络问题 前端请求后端接口发现得到的响应不对&#xff0c;或者通过 f12 明明看到了后端的响应&#xff0c;但是前端页面无法正常的展示出来。 这种情况该怎么排查呢&#xff1f;这篇我们就来好好的理一理&#xff01; 1、配置问题 1&am…

浅谈取样器插件之jp@gc - UDP Request

浅谈取样器插件之jpgc - UDP Request JPgc - UDP Request允许用户在性能测试中发送UDP&#xff08;User Datagram Protocol&#xff09;数据包。这对于测试那些依赖UDP协议进行通信的应用程序和服务特别有用&#xff0c;比如某些物联网&#xff08;IoT&#xff09;设备、在线游…

我在高职教STM32——I2C通信入门(2)

大家好,我是老耿,高职青椒一枚,一直从事单片机、嵌入式、物联网等课程的教学。对于高职的学生层次,同行应该都懂的,老师在课堂上教学几乎是没什么成就感的。正是如此,才有了借助CSDN平台寻求认同感和成就感的想法。在这里,我准备陆续把自己花了很多心思设计的教学课件分…

QT输入组、QT显示组

目录 QT输入组 ​编辑 Combo Box&#xff08;下拉菜单部件&#xff09; Font Combo Box&#xff08;显示系统中可用的字体&#xff09; Line Edit&#xff08;行编辑器&#xff09; Text Edit&#xff08;文本编辑器&#xff09; Plain Text Edit&#xff08;纯文本编辑…

干货实操分享:6个禁用外来u盘的方法

禁用外来U盘是保护计算机系统和数据安全的重要措施之一。以下是六个禁用外来U盘的方法&#xff0c;旨在帮助您有效防止未经授权的U盘接入计算机&#xff1a; 1. 使用组策略编辑器&#xff08;Windows系统&#xff09; 步骤&#xff1a; 打开“运行”对话框&#xff0c;输入g…

Covalent(CXT)与Sei合作,为扩展以太坊应用提供数据解决方案

Covalent Network&#xff08;CXT&#xff09;是领先的模块化数据基础设施层&#xff0c;致力于解决长期数据可用性&#xff0c;并为 AI 提供可验证的结构化数据。目前 Covalent Network&#xff08;CXT&#xff09;已经与首个并行化 EVM 区块链——Sei 达成了新的合作&#xf…

cf 练习3

cf 955 div.2 D (二维前缀和 裴蜀定理) 设原本 有雪帽的点&#xff08;设为1&#xff09; 和 没有学帽&#xff08;设为0&#xff09;的点 差值为 dif 当边长为k的矩阵覆盖后 &#xff0c; 设矩阵中有x1个有雪帽的点和 x2个没有雪帽的点 &#xff0c;那么此时的dif 值 就会减…

【建议收藏】AI大模型学习资源大全,免费分享

前言 马斯克旗下的AI大模型企业xAI&#xff0c;官宣拿到巨额融资60亿美元。 估值冲向240亿美元&#xff08;约1304亿&#xff09;&#xff0c;一举跃升为AI大模型“最猛独角兽”&#xff01; 资本大佬Valor Equity Partners、红杉资本、国王控股、沙特王子-Bin Talal、Vy Cap…

同态加密和SEAL库的介绍(四)CKKS 方案

写在前面&#xff1a; 上篇介绍了 BFV 的 Batch Encoder&#xff0c;其虽然充分利用了槽空间&#xff0c;但是每个槽只包含一个模 plain_modulus 的整数&#xff0c;除非 plain_modulus 非常大&#xff0c;否则我们可能会很快遇到数据类型溢出并在需要进行整数计算时得到意外的…

【Docker安装】Ubuntu系统下离线部署Docker环境教程

【Docker安装】Ubuntu系统下离线部署Docker环境教程 前言一、本次实践介绍1.1 本次实践规划1.2 本次实践简介二、检查本地环境2.1 检查操作系统版本2.2 检查内核版本2.3 更新软件源三、卸载Docker四、下载安装包4.1 创建目录4.2 官网下载五、部署Docker环境5.1 解压安装包5.2 复…

北京青蓝智慧科技:160个项目通过“数据要素×”大赛湖北分赛初赛

近日&#xff0c;2024年“数据要素”大赛的湖北分赛在武汉热烈开幕。 八个赛道的参赛队伍齐聚一堂&#xff0c;共同争夺数据创新先锋的殊荣。 经过激烈的角逐&#xff0c;初赛评审专家团最终评选出了160个入围项目&#xff0c;每个赛道分别有20个项目脱颖而出&#xff0c;其中…

Unity物理模块 之 2D效应器

本文仅作笔记学习和分享&#xff0c;不用做任何商业用途 本文包括但不限于unity官方手册&#xff0c;unity唐老狮等教程知识&#xff0c;如有不足还请斧正​ 1.什么是效应器 2D 效应器 - Unity 手册 2D 效应器是与 2D 碰撞器一起使用的组件&#xff0c;相当于预先编写好的插…