基础DP(初级版)
本文主要内容为基础DP,内容来源为《算法导论》,总结不易,转载请注明出处。
后续会更新出kuanbin关于基础DP的题目......
动态规划:
动态规划用于子问题重叠的情况,即不同的子问题具有相同的公共子子问题,在这种情况下分治算法会做许多不必要的工作,它会反复求解那些子子问题使得程序边的缓慢。而动态规划则对每个问题 只求解一次,将其解保存在一个表格中,从而避免一些不必要的重复计算。
动态规划常用来求解最优化问题,这类问题可以有很多解,每个解都有一个值,我们希望寻找具有最优值的解,我们称这样的解为问题的一个最优解,而不是最优解,因为可能有多个最优解。
动态规划设计算法的一般步骤:
1.刻画一个最优解的结构特征。
2.递归的定义最优解的值。
3.计算最优解的值,通常采用自地向上的方法。
4.利用计算出的信息构造一个最优解。
动态规划的两种基本解题步骤:
第一种为自顶向下法:此方法仍按自然的递归形式编写过程,但过程中会保存每个子问题的解。当需要一个子问题的解时,过程中会首先检查是否此问题已经被求解,如果是则直接返回该解,否则按通常的方式计算,我们称这个递归过程时带备忘的,因为他记住了之前的计算结果,不会进行重复的计算。
第二种为自底向上法:这种方法一般需要恰当定义子问题的规模的概念,使得任何子问题都只依赖更小的子问题求解。因而我们可以将子问题的规模排序按由小到大的顺序进行求解。当求解某个子问题时,它所依赖的更小的子问题已经得到解决,结果已经保存。每个子问题也只需求解一次。
最优子结构:
问题的最优解由相关子问题的最优解构成,这些子问题可以独立求解。
重构解:
在求解过程中保存相应的状态到另一个辅助数组中即可。
例:钢条切割问题:一根长度为n的钢条,切割不同的长度 i 对应不同的价格p[ i ], 问你如何切割一根钢条使得利益最大化。
n = i1 + i2 + ... + ik;
递推式:
rn = max(pn, r1 + r(n-1), r2 + r(n-2)...r(n-1) +r1)。
#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std; const int maxn = , INF = -0x3f3f3f3f;
long long dp[maxn], s[maxn];
long long p[maxn] = {, , , , , , , , , , };
long long q; long long memoized_cut_rod(int n) {
if(dp[n] >= ) return dp[n];
if(n == ) q = ;
else q = INF;
for(int i = ; i <= n; i ++)
q = max(q, p[i] + memoized_cut_rod(n - i));
dp[n] = q;
return q;
} long long bottom_up_cut_rod(int n) {
dp[] = ;
for(int j = ; j <= n; j ++) {
q = INF;
for(int i = ; i <= j; i ++) {
q = max(q, p[i] + dp[j - i]);
}
dp[j] = q;
}
return dp[n];
} int main () {
long long n;
memset(dp, -, sizeof dp);
while(cin >> n) {
long long ans = memoized_cut_rod(n);
printf("%d\n", ans);
ans = bottom_up_cut_rod(n);
printf("%d\n", ans);
}
}
重构解:
long long bottom_up_cut_rod(int n) {
dp[] = ;
for(int j = ; j <= n; j ++) {
q = INF;
for(int i = ; i <= j; i ++) {
if(p[i] + dp[j - i] > q) {
q = max(q, p[i] + dp[j - i]);
s[j] = i;
}
}
dp[j] = q;
}
return dp[n];
} while(n) {
cout << s[n] << '\t';
n = n - s[n];
}
例二:求斐波纳挈数
#include <iostream>
using namespace std; const int maxn = , INF = 0x3f3f3f3f;
long long dp[maxn]; long long calculate_fib(int n) {
if(n == && n == ) return ;
for(int i = ; i <= n; i ++)
if(dp[i] < ) dp[i] = dp[i - ] + dp[i - ];
return dp[n];
} int main () {
long long n;
for(int i = ; i < maxn; i ++) dp[i] = -INF;
dp[] = dp[] = ;
while(cin >> n) {
cout << calculate_fib(n);
}
}
基础DP(初级版)的更多相关文章
- NSIS安装制作基础教程[初级篇], 献给对NSIS有兴趣的初学者
NSIS安装制作基础教程[初级篇], 献给对NSIS有兴趣的初学者 作者: raindy 来源:http://bbs.hanzify.org/index.php?showtopic=30029 时间: ...
- [osg][osgEarth][原]基于OE自定义自由飞行漫游器(初级版)
由于受够了OE的漫游器,想搞个可以在全球飞行的漫游器,所以就做了一个: 请无视我的起名规则······ 类头文件:EarthWalkManipulator.h #pragma once //南水之源 ...
- 我的新书《计算机图形学基础(OpenGL版)》
我的新书<计算机图形学基础(OpenGL版)>今年6月份在清华大学出版社出版了!新书与原在机械工业出版社出的<计算机图形学>相比,主要有以下不同: 1.加重OpenGL的内容, ...
- 02 初级版web框架
02 初级版web框架 服务器server端python程序(初级版): import socket server=socket.socket() server.bind(("127.0.0 ...
- Python之路,Day4 - Python基础4 (new版)
Python之路,Day4 - Python基础4 (new版) 本节内容 迭代器&生成器 装饰器 Json & pickle 数据序列化 软件目录结构规范 作业:ATM项目开发 ...
- 基础dp
队友的建议,让我去学一学kuangbin的基础dp,在这里小小的整理总结一下吧. 首先我感觉自己还远远不够称为一个dp选手,一是这些题目还远不够,二是定义状态的经验不足.不过这些题目让我在一定程度上加 ...
- hdu 5586 Sum 基础dp
Sum Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) Problem Desc ...
- 希尔排序之C++实现(初级版)
希尔排序之C++实现(初级版) 一.源代码:希尔排序之C++实现(初级版) /*希尔排序基本思想: 先取一个小于n的整数d1作为第一个增量,把文件的全部记录分组. 所有距离为d1的倍数的记录放在同一个 ...
- 直接插入排序(初级版)之C++实现
直接插入排序(初级版)之C++实现 一.源代码:InsertSortLow.cpp /*直接插入排序思想: 假设待排序的记录存放在数组R[1..n]中.初始时,R[1]自成1个有序区,无序区为R[2. ...
随机推荐
- eclipse gradle插件 org.gradle.tooling.GradleConnectionException: Could not install Gradle distribution from 'https://services.gradle.org/distributions/gradle-3.4-bin.zip'.
eclipse安装gradle后出现如下异常: org.gradle.tooling.GradleConnectionException: Could not install Gradle distr ...
- Shell 编程(实例二)
创建一个脚本,为指定硬盘创建分区 1.列出当前系统所有磁盘,让用户选择,如果选择quit则退出脚本:如果选择错误,则重新选择 2.当用户选择完成后,询问是否继续 3.抹除选择磁盘上的分区,为其创建三个 ...
- Centos代理上网设置
原文链接:http://m.blog.csdn.net/article/details?id=51851677 一.centos自带界面设置代理 1. 界面设置 squid默认代理端口3128. 2 ...
- air报错 Error: Error #3000: Illegal path name
配置增加: <supportedProfiles>extendedDesktop desktop</supportedProfiles> fb: flash:
- JSP页面java代码报错:Purgoods cannot be resolved to a type
错误提示 : Purgoods cannot be resolved to a type Purgoods不能解析为一个类型 原因 : 缺少引入Purgoods类 页面中引入java类,执行java代 ...
- pycharm 直接删掉数据表之后,makemigration和migrate 之后,数据库中依然没有生成数据表的问题
综合分析一下行程这个问题的原因: 在终端中运行 select * from django_migrations; 查看 提交的记录,如果你的表删掉了,记录还在,那么数据库会觉得,这个表依然是存在的,所 ...
- python玩丢手绢问题,出局的顺序
# 丢手绢问题# 游戏规则: 有N个小朋友玩丢手绢游戏,做成一圈,从第一个小朋友开始数数,从一开始数,数到指定数字的小朋友要出列,然后下一个小朋友继续从1开始数,依次类推,算出最后一个留下来的小朋友是 ...
- 简单几步手工扩容LVM(笔记)
参考文档:https://www.cnblogs.com/einyboy/archive/2012/05/31/2528661.html 1.查看磁盘是否被系统认出: fdisk -l 如显示加的磁盘 ...
- [JS]如何理解JS中的类和对象
-------------------------------------------------------------------------------------------- 变量:自由的 ...
- jenkins 自动触发
在gitlab上配置连接jenkins ,将Jenkins的Secret token 与Build URL 复制到gitlab中 在settings标签下面,找到OutBound Request,勾选 ...