算法基础篇-10-动态规划

news2025/1/13 7:37:22

1. 简介

动态规划是一种算法思想; 动态规划=递归式+子问题

1.1 案例1: 斐波那契数列

  • 斐波那契数列:Fn=Fn-1+Fn-2

代码演示: 使用递归和非递归的方法来求解斐波那契数列的第n项;

递归的方法:

   public static int fbnc(int n) {
        if (n == 1 || n == 2) {
            return 1;
        }
        return fbnc(n - 1) + fbnc(n - 2);
    }

非递归的方法:

    public static Long fbnc2(int n) {
        ArrayList<Long> list = new ArrayList<>();
        list.add(0L);
        list.add(1L);
        list.add(1L);
        if (n > 2) {
            for (int i = 0; i < n - 2; i++) {
                long num = list.get(list.size() - 1) + list.get(list.size() - 2);
                list.add(num);
            }
        }
        return list.get(n);
    }

对比发现,递归其实相比之下很慢递归子问题重复计算,假设n=6,是否递归存在子问题的重复计算:

  • f(6)=f(5)+f(4)
    f(5)=f(4)+f(3)
    f(4)=f(3)+f(2)
    f(4)=f(3)+f(2)
    f(3)=f(2)+f(1)
    f(3)=f(2)+f(1)
    f(3)=f(2)+f(1)
    f(2)=1

1.2 案例2:钢条切割问题

  • 某公司出售钢条,出售价格与钢条长度之间的关系如下表:
    在这里插入图片描述
    问题:现在有一段长度为n的钢条和上面的价格表,求切割钢条方案,使得总收益最大;
    假设现在钢条长度是4米,那么所有切割的方案如下:
    在这里插入图片描述
    如上图所示,长度为4的时候,有8种切割方案,当长度为n的时候
    在这里插入图片描述

1.2.1 递推式:

长度为n的钢条切割后最优收益为rn,可以得出
在这里插入图片描述

  • 第一个参数Pn 表示不切割;
  • 其他n-1 个参数分别表示另外n-1种不同切割方案,对方案i=1,2,n-1
    • 将钢条切割为长度为 i 和 n-i 两段;
    • 方案i 的收益为切割两段的最优收益之和;
  • 考察所有的i,选择其中收益最大的方案;
  • 代码如下:
    public static void main(String[] args) {
        int[] arr = {0, 1, 5, 8, 9, 10, 17, 17, 20, 24, 30};
        System.out.println("getMaxValue(arr,9) = " + getMaxValue(arr, 9));
    }
    
    public static int getMaxValue(int[] arr, int n) {
        if (n == 0) {
            return 0;
        }else {
            int res = arr[n];
            for (int i = 1; i < n; i++) {
                res = Math.max(res, getMaxValue(arr, i) + getMaxValue(arr, n - i));
            }
            return res;
        }
    }

1.2.2 最优子结构

  • 可以将求解规模为n的原问题,划分为规模更小的子问题:完成依次切割后,可以将产生的两段钢条看成两个独立的钢条切割问题;
  • 组合两个子问题的最优解,并在所有可能的两段分割方案中选取组合收益最大的,构成原问题的最优解;
  • 钢条切割满足最优子结构,问题的最优解由相关子问题的最优解组合而成,这些子问题可以独立求解;
  • 钢条切割问题还存在更简单的递归求解方法
    • 从钢条的左边切割下长度为i的一段,只对右边剩下的一段继续进行切割,左边的不再切割;
    • 递推式简化为
      在这里插入图片描述
    • 不做切割的方案就可以描述为:左边一段长度为n,收益为pn,剩余一段长度为0,收益为r0=0
    • 代码如下所示:
    public static void main(String[] args) {
        int[] arr = {0, 1, 5, 8, 9, 10, 17, 17, 20, 24, 30};
        System.out.println("getMaxValue(arr,9) = " + getMaxValue(arr, 9));
    }

    public static int getMaxValue(int[] arr, int n) {
        if (n == 0) {
            return 0;
        } else {
            int res = 0;
            for (int i = 1; i < n + 1; i++) {
                res = Math.max(res, arr[i] + getMaxValue(arr, n - i));
            }
            return res;
        }
    }

1.2.3 自顶向下递归实现

  • 为什么自顶向下递归写出来的算法效率低下,因为存在子问题重复问题,时间复杂度是O(2^n)
    在这里插入图片描述
    由于递归算法重复求解相同子问题,效率低下;动态规划思想:每个子问题只求解一次,保存求解结果,之后需要此问题时,只需查找保存的结果即可;
    public static void main(String[] args) {
        int[] arr = {0, 1, 5, 8, 9, 10, 17, 17, 20, 24, 30};
        System.out.println("getMaxValue(arr,9) = " + getMaxValue(arr, 9));
    }
    
    public static int getMaxValue(int[] arr, int n) {
        if (n == 0) {
            return 0;
        }else {
            int res = arr[n];
            for (int i = 1; i < n; i++) {
                res = Math.max(res, getMaxValue(arr, i) + getMaxValue(arr, n - i));
            }
            return res;
        }
    }

1.2.4 自下而上实现

  • 时间复杂度O(n^2)
    在这里插入图片描述
    public static void main(String[] args) {
        int[] arr = {0, 1, 5, 8, 9, 10, 17, 17, 20, 24, 30};
        System.out.println("getMax(arr,9) = " + getMax(arr, 9));
    }


    public static int getMax(int[] arr, int n) {
        List<Integer> list = new ArrayList<>();
        list.add(0);
        for (int i = 1; i < n + 1; i++) {
            int res = 0;
            for (int j = 1; j < i + 1; j++) {
                res = Math.max(res, arr[j] + list.get(i - j));
            }
            list.add(res);
        }
        return list.get(n);
    }

1.2.5 获取最优解的切割方式

基于原来的代码,将具体的切割方式打印出来

 public static void main(String[] args) {
        int[] arr = {0, 1, 5, 8, 9, 10, 17, 17, 20, 24, 30};
        System.out.println("getMax(arr,9) = " + getMaxExtend(arr, 9));
    }
    
    public static List<Integer> getMaxExtend(int[] arr, int n) {
        List<Integer> result = new ArrayList<>();
        result.add(0);
        List<Integer> typeList = new ArrayList<>();
        typeList.add(0);

        for (int i = 1; i < n + 1; i++) {
            int res = 0;  //价格的最大值
            int type = 0;
            for (int j = 1; j < i + 1; j++) {
                if (arr[j] + result.get(i - j) > res) {
                    res = arr[j] + result.get(i - j);
                    type = j;
                }
            }
            result.add(res);
            typeList.add(type);
        }
        List<Integer> objects = new ArrayList<>();
        while (n > 0) {
            objects.add(typeList.get(n));
            n = n - typeList.get(n);
        }
        return objects;
    }

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

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

相关文章

uniapp实战仿写网易云音乐(二)—视频页面(scroll-view组件实现横线滑动,mescroll-uni实现视频列表,向下滑动刷新当前页面)

文章目录前言顶部导航条的实现视频列表的展示部分CSS部分最后前言 接着上篇文章继续完成剩下的部分&#xff0c;本篇文章是完成第二个页面——视频页面的部分&#xff0c;视频还是没有做播放的效果&#xff0c;主要是做展示效果。下面附上两篇文章链接&#xff0c;没看过的同学…

【概率论】正态分布

前导知识&#xff1a; 概率密度函数&#xff08;密度函数&#xff09;&#xff1a;描述一个随机变量的在某个确定的取值点附近的可能性的函数。 随机变量的取值落在某个区域内的概率为概率密度函数在这个区域上的积分。 性质&#xff1a; f(x)>0 数学期望 又称均值&am…

vue组件

9.Vue组件 组件&#xff08;Component&#xff09;是 Vue.js 最强大的功能之一。 组件可以扩展 HTML 元素&#xff0c;封装可重用的代码。 组件系统让我们可以用独立可复用的小组件来构建大型应用&#xff0c;几乎任意类型的应用的界面都可以抽象为一个组件树&#xff1a; 9…

SpringBoot 自定义starter yaml提示失效问题

自定义starter yaml提示失效问题自定义starter yaml提示失效问题问题场景解决办法自定义starter yaml提示失效问题 问题场景 在自定义starter后&#xff0c;必不可少会有properties配置参数需要指定&#xff0c;而在有时又不知道为什么出现这个问题。官方 Configuration Meta…

玩以太坊链上项目的必备技能(OOP-抽象合约-Solidity之旅十)

抽象合约&#xff08;abstract contract&#xff09; 前文在讲合约继承的基类构造函数的参数时&#xff0c;有提到抽象合约&#xff0c;也就是说&#xff0c;如果派生合约未能给其继承的基合约指定构造函数参数时&#xff0c;那么&#xff0c;该派生合约必须声明为抽象合约&am…

ubuntu中安装tippecanoe并切片

概述 本文是一片”水文”&#xff0c;记录一下如何在ubuntu中用tippecanoe制作矢量切片。 实现操作 本示例中ubuntu是在VMware虚拟机中&#xff0c;安装的是18.04.6的版本&#xff0c;你可通过我兰的镜像下载&#xff0c;速度杠杠的。 1.安装git sudo apt install git2.cl…

探索SpringMVC-HandlerMapping之RequestMappingHandlerMapping

前言 上回我们知道HandlerMapping是用来寻找Handler的&#xff0c;并不与Handler的类型或者实现绑定&#xff0c;而是根据需要定义的。那么为什么要单独给RequestMapping实现一个HandlerMapping&#xff1f;这次咱们就来专门看看这个RequestMappingHandlerMapping。 RequestM…

Educational Codeforces Round 139 (Rated for Div. 2)(A~E)

A. Extremely Round 定义一个数中仅存在一位非0&#xff0c;它是extremely round&#xff0c;计算1~n中有几个满足条件的数。 思路&#xff1a;直接计算即可。 AC Code&#xff1a; #include <bits/stdc.h>typedef long long ll; const int N 1e5 5; int t, n;int c…

高手必备10大难题:Mysql如何实现RR级隔离时,不会幻读?

文章很长&#xff0c;而且持续更新&#xff0c;建议收藏起来&#xff0c;慢慢读&#xff01;疯狂创客圈总目录 博客园版 为您奉上珍贵的学习资源 &#xff1a; 免费赠送 :《尼恩Java面试宝典》 持续更新 史上最全 面试必备 2000页 面试必备 大厂必备 涨薪必备 免费赠送 经典…

[2022-12-17]神经网络与深度学习第5章 - 循环神经网络(part 1)

contents循环神经网络 part 1 - RNN记忆能力实验写在开头循环神经网络的记忆能力实验数据集构建数据集构建函数数据集加载构建 Dataset类模型构建嵌入层SRN层自己实现torch框架实现比较线性层模型汇总模型训练训练指定长度的数字预测模型模型评价写在最后循环神经网络 part 1 -…

[机器人学习]-树莓派6R机械臂运动学分析

根据D-H表规定得到如下变换矩阵为&#xff1a; 由此可得机器人相邻两关节位姿分别为&#xff1a; 根据DH参数求解变换矩阵的函数trans&#xff1a; %输入JD&#xff0c;即6个关节变量的值&#xff0c;求解正运动方程 function [ T ] trans( theta, d, a, alpha ) T [ cos(th…

Java中Stream的 flatMap 与 map 实际使用场景与区别对比

基本概念 Stream 流式操作&#xff0c;一般用于操作集合即 List 一类的数据结构&#xff0c;JDK 1.8 后的新特性 Stream 中的 map 一般用于对List 中的每一个元素执行指定方法使得最终结果为最终的集合为每一个记录的某一属性的集合&#xff08;get 方法&#xff09;或者通过…

PPT怎么转PDF?将Powerpoint(PPT)转换为PDF方法分享

当您在PowerPoint中创建精美的演示文稿时&#xff0c;您不仅想给观众留下深刻的印象&#xff0c;还希望它能够方便地打包&#xff0c;并且文件格式起着重要作用。虽然PPTX是一种广泛传播的格式&#xff0c;但PDF始终是一个安全的选择。以下是有关如何使用在线工具将PowerPoint演…

每天五分钟机器学习:主成分分析算法PCA的降维原理是什么?

本文重点 当数据从n维降到k维的时候,我们需要找到一个能使得投影误差最小的k维向量构成的投影平面,然后将数据进行投影,从而达到降维的作用。所以PCA算法要做的就是两件事情,一件事情就是计算最优的k维向量,另一个问题就是计算原来样本点映射到降维面的z(i),也就是这些新…

【C++初阶】模板初阶、STL简介

文章目录模板初阶泛型编程函数模板类模板STL简介什么是STLSTL的六大组件STL的缺陷模板初阶 泛型编程 概念 泛型编程&#xff1a;编写与类型无关的通用代码&#xff0c;是代码复用的一种手段。模板是泛型编程的基础。 函数模板 概念 函数模板代表了一个函数家族&#xff0c;…

搜索与图论 - spfa 算法

文章目录一、spfa 算法1. spfa 算法简介2. spfa 算法和 bellman-ford 算法的区别3. spfa 算法和 dijkstra 算法的区别4. spfa 算法实现步骤5. spfa 算法举例图解6. spfa 算法用于求最短路和判断负环&#xff0c;详见下面两道例题。二、spfa 算法例题—— spfa 求最短路具体实现…

1. SpringBoot 整合 Canal

勿以恶小而为之&#xff0c;勿以善小而不为----- 刘备 SpringBoot 整合 Canal pom.xml 添加 canal.client 依赖 (1.1.5 改动很大&#xff0c;这儿客户端用 1.1.4) <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.a…

【云原生进阶之容器】第一章Docker核心技术1.5.3节——cgroups数据结构剖析

1 cgroups数据结构解析 从进程的角度出发来剖析 cgroups 相关数据结构之间的关系。在 Linux 中管理进程的数据结构是 task_struct。cgroup表示进程的行为控制,因为子系统必须要知道进程是位于哪一个cgroup,所以在struct task_struct和cgroup中存在一种映射。 1.1 task_struc…

【数据结构进阶】二叉平衡树

一、 二叉平衡树 概念 二叉搜索树有称 二叉排序树&#xff0c;它也可以是一个空树。 如果它的左子树不为空&#xff0c;则左子树上所有结点的值都小于根结点的值如果他的右子树不为空&#xff0c;则右子树上所有结点的值都大于根结点的值它的左右子树也分别是二叉搜索树 由…

【Acwing 周赛#82】AcWing 4783. 多米诺骨牌

目录 4782.第k个数 4783. 多米诺骨牌 - bfs 记录时间 4782.第k个数 java大顶堆 import java.util.*;public class Main {public static void main(String[] args ){Scanner scnew Scanner(System.in);int nsc.nextInt(),ksc.nextInt();k--;PriorityQueue<Integer> qn…