dp入门 石子相邻合并 详细带图讲解
题目:
有N堆石子,现要将石子有序的合并成一堆,规定如下:
1.每次只能移动相邻的2堆石子合并
2.合并花费为新合成的一堆石子的数量。
求将这N堆石子合并成一堆的总花费最小(或最大)。
样例:
输入:7
13 7 8 16 21 4 18
输出:239
说是简单dp,刚开始学dp还是有点困难,这个题目我花了很长时间了。今天差不多才理清里面的大部分细节。
首先,合并相邻的石子,合并时,可以两堆,三堆,四堆。两堆的时候就只能合并相邻两堆;三堆的时候就有多种选择了,1+2 和 2+1;四堆的时候,1+3 和 2+2 和 3+1;
2堆的时候:
3堆的时候:

4堆也是类似。
现在我们拿样例来分析一下。
1.首先是状态转移方程。
设f[i][j]是从第i堆到第j堆的最优值。 配和上面的图,两堆的时候:f[1][2] f[2][3] f[3][4] f[4][5] f[5][6] f[6][7]; 三堆的时候:f[1][3] f[2][4] ......。
然后 f[1][3] = f[1][1] + f[2][3] + s[3] - s[0]; f[1][3] = f[1][2] + f[3][3] + s[3] - s[0]; 有这两种情况。取他们的最小值。
转移方程就是 f[i][j] = min( f[i][j], f[i][k] + f[k+1][j] +s[j]-s[i]);
2.为什么是s[3] - s[0]呢?
s[3] - s[0]是从第1堆到第3堆的总和。但是花销并不仅仅是总和, 合并3堆的时候,其实是需要合并两次,所以每堆都用到了两次,而自己和自己合并,花费的价值为0。f[1][3] = f[1][1] + f[2][3] + s[3] - s[0]; f[1][1] + f[2][3]这个是一次合并,s[3] - s[0]这个是另外一次合并。
3.输出的答案是f[1][n]。
代表从第1堆到第n堆的最小花费。
4.为什么要倒推?
因为顺推的时候,f[1][3] = f[2][3] + f[1][1] ,顺推的话,i从1开始,那f[2][3]这个时候是没办法知道的。所以倒推。
代码:
#include<bits/stdc++.h>
using namespace std;
const int M = 10009; int f[M][M]; // f[i][j] 代表第 i 堆石子 到 第 j 堆石子的最优值
int x,s[M]; int main(){
int n;
cin>>n;
memset(f,1011/3,sizeof(f));
for(int i = 1; i <= n; i++)
{
cin>>x;
s[i] = s[i-1] + x; //s[i] 代表前 i 堆石子的总和
} for(int i = 1; i <= n; i++) f[i][i] = 0; for(int i = n-1; i >= 1; i--)
for(int j = i+1; j <= n; j++)
for(int k = i; k <= j-1; k++)
{
f[i][j] = min(f[i][j],f[i][k]+f[k+1][j]+s[j]-s[i-1]);
} cout<<f[1][n]; return 0;
}
dp入门 石子相邻合并 详细带图讲解的更多相关文章
- DP之石子堆合并问题
相邻 环形 总结 (1)相邻:在一个圆形操场的四周摆放着n堆石子(n<= 100),现要将石子有次序地合并成一堆.规定每次只能选取相邻的两堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得 ...
- HRBUST - 1818 石子合并 区间dp入门
有点理解了进阶指南上说的”阶段,状态和决策“ /* 区间dp的基础题: 以区间长度[2,n]为阶段,枚举该长度的区间,状态dp[l][r]表示合并区间[l,r]的最小费用 状态转移方程dp[l][r] ...
- IT技术学习指导之Linux系统入门的4个阶段(纯干货带图)
IT技术学习指导之Linux系统入门的4个阶段(纯干货带图) 全世界60%的人都在使用Linux.几乎没有人没有受到Linux系统的"恩惠",我们享受的大量服务(包括网页服务.聊天 ...
- wyh的dp入门刷题笔记
0: 靠前感觉之前dp抄题解都是抄的题解,自己从没有真正理解过dp.wyh下了很大决心从头学dp,于是便有了这篇文章. 1.背包 前四讲01背包&多重背包&完全背包(混合背包) :樱花 ...
- 能量项链(区间DP入门)
题面:能量项链https://www.luogu.com.cn/problem/P1063 乍一看和石子合并差不多,可是多了头值和尾值,看起来十分麻烦 我们画一张图,紫色表示头值,蓝色表示尾值.规定西 ...
- poj 3254 状压dp入门题
1.poj 3254 Corn Fields 状态压缩dp入门题 2.总结:二进制实在巧妙,以前从来没想过可以这样用. 题意:n行m列,1表示肥沃,0表示贫瘠,把牛放在肥沃处,要求所有牛不能相 ...
- 数位dp入门 hdu2089 不要62
数位dp入门 hdu2089 不要62 题意: 给定一个区间[n,m] (0< n ≤ m<1000000),找出不含4和'62'的数的个数 (ps:开始以为直接暴力可以..貌似可以,但是 ...
- poj3254状压DP入门
G - 状压dp Crawling in process... Crawling failed Time Limit:2000MS Memory Limit:65536KB 64bit ...
- Spark+ECLIPSE+JAVA+MAVEN windows开发环境搭建及入门实例【附详细代码】
http://blog.csdn.net/xiefu5hh/article/details/51707529 Spark+ECLIPSE+JAVA+MAVEN windows开发环境搭建及入门实例[附 ...
随机推荐
- day 0308 编码的进阶 文件操作
一.编码的进阶: 在python3以后,字符串和bytes类型彻底分开,字符串以字符为单位进行处理的,bytes类型是以字节为单位处理的. bytes数据类型在所有的操作和使用与字符串方法基本一样,也 ...
- [development][http][libhtp] suricata的http库--libhtp
首先,从文档来看, 它支持管道化的http, 也可以说过于重量级. 其次, 还没有达到产品化的稳定性. 并不是完全对应我的需求, http模块是一个整体性能相关模块. 应该按需定制更合理. 但是,功能 ...
- linux screen工具
https://www.cnblogs.com/lpfuture/p/5786843.html 一.背景 系统管理员经常需要SSH 或者telent 远程登录到Linux 服务器,经常运行一些需要很长 ...
- 抽屉之Tornado实战(5)--点赞与评论树
点赞 点赞的过程:数字增加,并在后台点赞表记录数据 需要发过去的数据:用户id,新闻id 用户id从session里获得,那新闻id怎么获取呢?这想到分页是循环新闻列表来展示内容,循环的新闻id可以做 ...
- 《mongoDB》查询
一:简单查询 db.collection.find(query, projection) - query :可选,使用查询操作符指定查询条件 - projection :可选,使用投影操作符指定返回的 ...
- Java之旅_面向对象_抽象类
参考并摘自:http://www.runoob.com/java/java-abstraction.html Java抽象类: 在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有 ...
- idea的基本使用
对于Idea没有workspace的概念,但是它把每个项目都看成是maven的一个模块,在一个idea窗口要想显示多个项目时就和eclipse不太一样,下面会详细介绍. 另外maven的setting ...
- 使用反向代理的http的请求流程
此文章主要为刚接触反向代理的小伙伴梳理请求流程,以便更好的理解反向代理是何时工作的 流程 由于浏览器是有缓存的,所以本地的hosts文件的信息也会在浏览器端缓存 当客户端发起一个新的请求(例如:输入的 ...
- 7620N路由器刷openwrt系统并配置***
之前一直用goAgent***,主要用gmail.逛逛twitter.youtube.维基百科.加速stackoverflow等等.但goAgent只能PC使用,手机.iPad都不能用,直到我拿到了一 ...
- uarts裸机程序
硬件平台:JZ2440 实现功能:向串口软件实现输出putchar函数 start.s --> 设置堆栈,关闭看门狗,初始化时钟,初始化sdram init.c -->初始化 ...