我的编程空间,编程开发者的网络收藏夹
学习永远不晚

c++动态规划经典算法

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

c++动态规划经典算法

基本思想

         动态规划算法通常用于求解具有某种最优性质的问题。在这类问题中,可能会有许多可行解。每一个解都对应于一个值,我们希望找到具有最优值的解。动态规划算法与分治法类似,其基本思想也是将待求解问题分解成若干个子问题,先求解子问题,然后从这些子问题的解得到原问题的解。与分治法不同的是,适合于用动态规划求解的问题,经分解得到子问题往往不是互相独立的。若用分治法来解这类问题,则分解得到的子问题数目太多,有些子问题被重复计算了很多次。如果我们能够保存已解决的子问题的答案,而在需要时再找出已求得的答案,这样就可以避免大量的重复计算,节省时间。我们可以用一个表来记录所有已解的子问题的答案。不管该子问题以后是否被用到,只要它被计算过,就将其结果填入表中。这就是动态规划法的基本思路。具体的动态规划算法多种多样,但它们具有相同的填表格式。

重要分析问题方法

        动态规划算法跟数组有着密切的关系,因此推荐大家在分析动态规划的算法时画一张表格(建议使用excel)分析解决问题往往能够事半功倍。

动态规划算法实例

1、台阶问题

 问题描述:有10级台阶,一个人每次上一级或者两级,问有多少种走完10级台阶的方法。

结合表格分析问题,每个子问题都只跟台阶这个变量相关,可以画出一个一维表格进行分析。

  1 2 3 4 5 6 7 8 9 10
走法 1 2 3 5 8 13 21 34 55 89

分析边界条件:对于每个台阶每次上一级或者两级,当台阶数为1时走法为1(即走一级即毕)为2时走法为2(走两次一级和走一次二级)。

分析递归关系:对于任一台阶都可以分为通过两级或者一阶到达。

      

终于,在有了上述两个条件之后,问题轻易得到了求解。(结合表格分析的手段到这里还没有完全发挥出它该有的优势,大家拭目以待)

台阶问题基于c++的代码实现:


// ConsoleApplication2.cpp : 定义控制台应用程序的入口点。
//
//台阶问题:有一座高度是10级台阶的楼梯,从下往上走,每跨一步只能向上1级或者2级台阶。要求用程序来求出一共有多少种走法。
#include "stdafx.h"
#include <iostream>
using namespace std;
int getResultByDP(int n)//自底向上的问题解法
{
	if (n<1)
	{
		return 0;
	}
	if (n==1)
	{
		return 1;
	}
	if (n==2)
	{
		return 2;
	}
	int a = 1;//从两个递归基开始
	int b = 2;
	int temp = 0;
	for (int i = 3; i < n + 1; i++)
	{
		temp = a + b;
		a = b;
		b = temp;
	}
	return temp;
}
int _tmain(int argc, _TCHAR* argv[])
{
	cout << getResultByDP(10);
	system("pause");
	return 0;
}

2、从矩阵左上角走到右下角最短路径问题

问题描述:给定一个矩阵m,从左上角开始每次只能向右走或者向下走,最后达到右下角的位置,路径中所有数字累加起来就是路径和,返回所有路径的最小路径和,如果给定的m如下,那么路径1,3,1,0,6,1,0就是最小路径和,返回12.

1 3 5 9

8 1 3 4

5 0 6 1

8 8 4 0

矩阵从左上角走到右下角
  0 1 2 3 4
0 0 0 0 0 0
1 0 1 3 5 9
2 0 8 1 3 5
3 0 5 0 6 1
4 0 8 8 4 0
           
  0 1 2 3 4
0 0 0 0 0 0
1 0 1 4 9 18
2 0 9 5 8 13
3 0 14 5 11 12
4 0 22 13 15 12

边界条件分析:由问题知道对于任一矩阵中的元素而言,上次位置或者是在该元素的坐标或者上边。那么当一些元素没有左边或者上边时应该怎么做呢?不妨就说的更为具体一些吧,如上图的表格所示当x(表示行下标)等于1,和y(表示列下标)等于1时正好是对应没有上边元素和没有左边元素的情况。对于只有左边元素的值array[x][y]=array[x][y-1]+m[x][y],对于只有上边元素:array[x][y]=array[x-1][y]+m[x][y](array为下面统计问题结果的二维数组,m为包含输入矩阵信息的二维数组)。

递归公式:对于平凡的子问题而言 (推导递归公式时刻意的考察array[x][y]和array[x-1][y]与array[x][y-1]的实际关系)

对于此问题而言:arry[x][y]=min(array[x-1][y],array[x][y-1])+m[x][y]

以下是该问题基于c++的代码实现:


//给定一个矩阵m,从左上角开始每次只能向右走或者向下走,最后达到右下角的位置,路径中所有数字累加起来就是路径和,返回所有路径的最小路径和,如果给定的m如下,那么路径1,3,1,0,6,1,0就是最小路径和,返回12.
#include "stdafx.h"
#include <string>
#include <iostream>
#include <algorithm>
using namespace std;
int const x_length=5, y_length=5;
int m[x_length][y_length] = {
	0, 0, 0, 0, 0,
	0, 1, 3, 5, 9,
	0, 8, 1, 3, 5,
	0, 5, 0, 6, 1,
	0, 8, 8, 4, 0
};
int minDis() //m二级指针(可以是一个二维数组)
{
	int dp[4 + 1][4 + 1];
	//---------初始化边界条件-----------------
	for (size_t i = 0; i < x_length; i++)
	{
		dp[i][0] = 0;
	}
	for (size_t j = 0; j < y_length; j++)
	{
		dp[0][j] = 0;
	}
	//-------------------------------------------
	for (size_t i = 1; i < x_length; i++)
	{
		for (size_t j= 1; j < y_length; j++)
		{
			if (i == 1)
			{
				dp[i][j] = dp[i][j - 1] + m[i][j];
			}
			else if (j == 1)
			{
				dp[i][j] = dp[i - 1][j] + m[i][j];
			}
			else
			{
				int temp1 = dp[i - 1][j] + m[i][j];
				int temp2 = dp[i][j - 1] + m[i][j];
				dp[i][j] = min(temp1, temp2);
			}			
		}
	}
	return dp[x_length - 1][y_length - 1];
}
int _tmain(int argc, _TCHAR* argv[])
{
	cout << "最右下角的最短路径为:" << minDis();
	system("pause");
	return 0;
}

3、最大子数组问题

问题描述:给定数组arr,返回arr的最长递增子序列的长度,比如arr=[2,1,5,3,6,4,8,9,7],最长递增子序列为[1,3,4,8,9]返回其长度为5,由于该问题中总要把当前元素和在他之前的进行分析,我们也是借助表格来直观的分析该问题。

    2 4 5 3 1    
    0 1 2 3 4 5 6
2 0              
4 1              
5 2              
3 3              
1 4              
                 
0 1 2 3 4        
1 2 3 2 1        

边界条件:显然对于第一个数而言有dp[0]=1(dp表示存放结果的数组)

递归公式:首先生成dp[n]的数组,dp[i]表示以必须arr[i]这个数结束的情况下产生的最大递增子序列的长度。对于第一个数来说,很明显dp[0]为1,当我们计算dp[i]的时候,我们去考察i位置之前的所有位置,找到i位置之前的最大的dp值,记为dp[j](0=<j<i),dp[j]代表以arr[j]结尾的最长递增序列,而dp[j]又是之前计算过的最大的那个值,我们在来判断arr[i]是否大于arr[j],如果大于dp[i]=dp[j]+1.计算完dp之后,我们找出dp中的最大值,即为这个串的最长递增序列。

该问题基于c++的代码实现:


//给定数组arr,返回arr的最长递增子序列的长度,比如arr=[2,1,5,3,6,4,8,9,7],最长递增子序列为[1,3,4,8,9]返回其长度为5.
#include "stdafx.h"
#include <string>
#include <iostream>
#include <algorithm>
#include <vector>
using namespace std;
int dp[5] = {};
int _tmain(int argc, _TCHAR* argv[])
{
 int arr[5] = { 2, 4, 5, 3, 1 };
 dp[0] = 1;
 const int oo = 0;
 for (int i = 1; i<5; i++){
  int _max = oo;
  for (int j = 0; j<i; j++)
   if (dp[j]>_max && arr[i]>arr[j])
    _max = dp[j];
  dp[i] = _max + 1;
 }
 int maxlist = 0;
 for (int i = 0; i < 5; i++)
  if (dp[i] > maxlist)
   maxlist = dp[i];
 cout << maxlist << endl;
 system("pause");
 return 0;
}

4、最长公共子序列

问题描述:给定两个字符串str1和str2,返回两个字符串的最长公共子序列,例如:str1="1A2C3D4B56",str2="B1D23CA45B6A","123456"和"12C4B6"都是最长公共子序列,返回哪一个都行。

问题分析:首先生成dp[n]的数组,dp[i]表示以必须arr[i]这个数结束的情况下产生的最大递增子序列的长度。对于第一个数来说,很明显dp[0]为1,当我们计算dp[i]的时候,我们去考察i位置之前的所有位置,找到i位置之前的最大的dp值,记为dp[j](0=<j<i),dp[j]代表以arr[j]结尾的最长递增序列,而dp[j]又是之前计算过的最大的那个值,我们在来判断arr[i]是否大于arr[j],如果大于dp[i]=dp[j]+1.计算完dp之后,我们找出dp中的最大值,即为这个串的最长递增序列。

    B D C A B A  
    0 1 2 3 4 5  
A 0 0 0 0 0 0 0 0
B 1 0 0 0 0 1 1 1
C 2 0 1 1 1 1 2 2
B 3 0 1 1 2 2 2 2
D 4 0 1 1 2 2 3 3
A 5 0 1 2 2 2 3 3
B 6 0 1 2 2 3 3 4
  7 0 1 2 2 3 4 4
                 
    B D C A B A  
    0 1 2 3 4 5  
A 0 -2 -2 -2 -2 -2 -2 -2
B 1 -2 -1 -1 -1 0 1 0
C 2 -2 0 1 1 -1 0 1
B 3 -2 -1 -1 0 1 -1 -1
D 4 -2 0 -1 -1 -1 0 1
A 5 -2 -1 0 -1 -1 -1 -1
B 6 -2 -1 -1 -1 0 -1 0
  7 -2 0 -1 -1 -1 0 -1

该问题基于c++代码实现:


//输入为两个长度不为零的字符串,输出这两个字符串的最大公共子序列
#include "stdafx.h"
#include <string>
#include <iostream>
#ifndef MAX
#define MAX(X,Y) ((X>=Y)? X:Y)
#endif
using namespace std;
int **Lcs_length(string X, string Y, int **B)
{
 int x_len = X.length();
 int y_len = Y.length();
 int **C = new int *[x_len + 1];
 for (int i = 0; i <= x_len; i++)
 {
  C[i] = new int[y_len + 1];        //定义一个存放最优解的值的表;
 }
 for (int i = 0; i <= x_len; i++)
 {
  C[i][0] = 0;
  B[i][0] = -2;                     //-2表示没有方向
 }
 for (int j = 0; j <= y_len; j++)
 {
  C[0][j] = 0;
  B[0][j] = -2;
 }
 for (int i = 1; i <= x_len; i++)
 {
  for (int j = 1; j <= y_len; j++)
  {
 
   if (X[i - 1] == Y[j - 1])
   {
    C[i][j] = C[i - 1][j - 1] + 1;
 
    B[i][j] = 0;             //0表示斜向左上
   }
   else
   {
    if (C[i - 1][j] >= C[i][j - 1])
    {
     C[i][j] = C[i - 1][j];
     B[i][j] = -1;       //-1表示竖直向上;
    }
    else
    {
     C[i][j] = C[i][j - 1];
     B[i][j] = 1;        //1表示横向左
    }
   }
 
  }
 }
 return C;
}
 
void OutPutLCS(int **B, string X, int str1_len, int str2_len)
{
 if (str1_len == 0 || str2_len == 0)
 {
  return;
 }
 if (B[str1_len][str2_len] == 0)   //箭头斜向左上
 {
  OutPutLCS(B, X, str1_len - 1, str2_len - 1);
  cout << X[str1_len - 1] << endl;
 }
 else if (B[str1_len][str2_len] == -1)
 {
  OutPutLCS(B, X, str1_len - 1, str2_len);
 }
 else
 {
  OutPutLCS(B, X, str1_len, str2_len - 1);
 }
}
 
int _tmain(int argc, _TCHAR* argv[])
{
 string X = "ABCBDAB";
 string Y = "BDCABA";
 
 int x_len = X.length();
 int y_len = Y.length();
 
 int **C;//定义一个二维数组
 
 int **B = new int *[x_len + 1];
 for (int i = 0; i <= x_len; i++)
 {
  B[i] = new int[y_len + 1];
 }
 C = Lcs_length(X, Y, B);
 for (int i = 0; i <= x_len; i++)
 {
  for (int j = 0; j <= y_len; j++)
  {
   cout << C[i][j] << " ";
  }
  cout << endl;
 }
 cout << endl;
 for (int i = 0; i <= x_len; i++)
 {
  for (int j = 0; j <= y_len; j++)
  {
   cout << B[i][j] << " ";
  }
  cout << endl;
 }
 OutPutLCS(B, X, x_len, y_len);//构造最优解
 system("pause");
 return 0;
}

到此这篇关于c++动态规划经典算法的文章就介绍到这了,更多相关c++动态规划内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

c++动态规划经典算法

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

C++动态规划算法如何使用

这篇“C++动态规划算法如何使用”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C++动态规划算法如何使用”文章吧。Fibon
2023-06-29

c语言动态规划算法是什么

C语言动态规划算法是一种用于解决优化问题的算法。它通过将问题划分为子问题,并保存子问题的解来避免重复计算,从而提高算法的效率。动态规划算法通常使用一个数组来保存子问题的解,这个数组称为“动态规划表”。算法的核心思想是通过填充动态规划表来逐步
2023-08-18

python动态规划算法怎么用

小编给大家分享一下python动态规划算法怎么用,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!python有哪些常用库python常用的库:1.requesuts;2.scrapy;3.pillow;4.twisted;5
2023-06-14

C++ 递归函数在动态规划算法中的应用?

动态规划算法中使用递归函数可以有效解决最优化问题。示例是斐波那契数列求解,递归函数基于公式 f(n) = f(n-1) + f(n-2)。可以通过使用备忘录技术优化递归函数,将子问题解决方案存储起来,避免重复计算。备忘录技术示例 is 创建
C++ 递归函数在动态规划算法中的应用?
2024-04-24

怎么使用C++动态规划算法实现矩阵链乘法

这篇文章主要介绍“怎么使用C++动态规划算法实现矩阵链乘法”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“怎么使用C++动态规划算法实现矩阵链乘法”文章能帮助大家解决问题。问题描述:给定n个矩阵的链<
2023-07-02

python实现动态规划算法的示例代码

本文主要介绍了python实现动态规划算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-02-16

怎么使用C++动态规划计算最大子数组

本文小编为大家详细介绍“怎么使用C++动态规划计算最大子数组”,内容详细,步骤清晰,细节处理妥当,希望这篇“怎么使用C++动态规划计算最大子数组”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。例题题目:输入一个整形
2023-07-02

Java算法之BFS,DFS,动态规划和贪心算法的实现

广度优先搜索(BFS)和深度优先搜索(DFS)是图遍历算法中最常见的两种算法,主要用于解决搜索和遍历问题。动态规划和贪心算法则用来解决优化问题。本文就来看看这些算法的具体实现吧
2023-05-14

C++实现动态规划过程详解

动态规划是解决一类最优问题的常用方法,它是解决最优化问题的一种途径,在本文中,我们将讨论如何使用C++实现动态规划算法,并提供一些示例来帮助您更好地理解该算法
2023-05-20

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录