【算法设计】动态规划算法设计——天平平衡、数塔问题(C++实现)

2023-10-25 16:36

本文主要是介绍【算法设计】动态规划算法设计——天平平衡、数塔问题(C++实现),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

创作不易,本篇文章如果帮助到了你,还请点赞 关注支持一下♡>𖥦<)!!
主页专栏有更多知识,如有疑问欢迎大家指正讨论,共同进步!
更多算法分析与设计知识专栏:算法分析🔥
给大家跳段街舞感谢支持!ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ

在这里插入图片描述


目录

  • 一、天平平衡问题
    • 问题描述
    • 算法思想和解题思路
    • C++代码
  • 二、数塔问题
    • 问题描述
    • 算法思想和解题思路
    • C++代码

一、天平平衡问题

问题描述

已知一个天平左右两端共有n个挂钩,且有m个不同质量的钩码,求将钩码全部挂到钩子上使天平平衡的方法的总数。试设计求解该问题的动态规划算法。

算法思想和解题思路

m个钩码要么挂在左边,要么挂在右边,使得左右平衡,也就是说使得左右两边钩码重量之和相等

左边或者右边的钩码重量之和是全部钩码重量之后的二分之一,天平的两端的重量和为m

从给定的数字数组中选择m个数字,并将它们分配到天平的两边,使得左右两边的数字之和相等

判断当前钩码的重量是否大于等于j,如果是的话,就将dp[j]加上dp[j - 当前钩码的重量]

	dp[j] = dp[j] + dp[j - arr[i - 1]];
  • 定义状态:dp[i]表示总重量为i时可以形成的平衡重量组合的数量。然后通过遍历每个钩码,尝试将这个钩码放到左边或者不使用这个钩码,然后更新dp数组
  • 状态转移方程:当前钩码的重量是否大于等于j,如果是则 dp[j] = dp[j] + dp[j - arr[i - 1]]; // 将dp[j]加上dp[j - 当前钩码的重量]
  • 边界条件:当没有钩码时,只有一种方案,即dp[0] = 1

在这里插入图片描述

C++代码

#include <iostream>
using namespace std;void sumMethod(int m,int n,int arr[])
{int dp[100] = { 0 };dp[0] = 1;for (int i = 1; i <= n; i++)		// 对钩码数量n进行循环  {for (int j = m; j >=1; j--)	// 对全部钩码的重量之和的二分之一m进行循环  {if (j >= arr[i - 1])		// 如果j大于等于当前钩码的重量  {dp[j] = dp[j] + dp[j - arr[i - 1]];	// 则将dp[j]加上dp[j - 当前钩码的重量]}}}for (int j = 0; j <= m; j++)	// 对全部钩码的重量之和的二分之一m进行循环  {  cout << dp[j] << " ";}cout << endl;
}int main()
{int m = 27;//全部钩码的重量之和的二分之一,问题中的nint n = 9;//钩码的数量,即题目中的m(个钩码)int a[] = { 10,9,8,7,6,5,4,3,2 };sumMethod(m, n, a);int m2 = 10;int n2 = 4;int a2[] = { 7,6,4,3 };sumMethod(m2, n2, a2);return 0;
}

在这里插入图片描述

二、数塔问题

问题描述

对于诸如下图的数塔,若从顶层走到底层,每一步只能走到相邻的结点,求经过的结点的数字之和最大的路径

试设计求解该问题的动态规划算法

在这里插入图片描述

算法思想和解题思路

从数塔的顶部开始,对于每个位置,选择向左走还是向右走,使得走过的路径上的数字之和最大

在数塔问题中,我们可以用一个二维数组dp来保存每个位置的最大值

  • 定义状态:data[i][j]表示数塔中位置(i, j)上的数字,对于每个位置(i, j),我们可以通过比较从左侧位置(i+1, j)和右侧位置(i+1, j+1)的最大值,来决定是从左走还是从右走
  • 状态转移方程:dp[i][j] = max(dp[i+1][j], dp[i+1][j+1]) + data[i][j]
  • 边界条件:dp[0][0]为数塔底部位置的最大值。

C++代码

#include <iostream>
#include <string.h>
#include <algorithm>
using namespace std;
void maxSum(int n)
{int arr[100][100];int dp[100][100];cout << "输入数塔:" << endl;for (int i = 0; i < n; i++){for (int j = 0; j <= i; j++){cin >> arr[i][j];dp[i][j] = arr[i][j];}}for (int i = n - 2; i >= 0; i--){for (int j = 0; j <= i; j++){dp[i][j] += max(dp[i + 1][j], dp[i + 1][j + 1]);}}cout << "数字之和最大为:" << dp[0][0] << endl;
}
int main()
{int n;cout << "输入数塔层数:";cin >> n;maxSum(n);return 0;
}

在这里插入图片描述


在这里插入图片描述

大家的点赞、收藏、关注将是我更新的最大动力! 欢迎留言或私信建议或问题。
大家的支持和反馈对我来说意义重大,我会继续不断努力提供有价值的内容!如果本文哪里有错误的地方还请大家多多指出(●'◡'●)

这篇关于【算法设计】动态规划算法设计——天平平衡、数塔问题(C++实现)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/283818

相关文章

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�