石子合并(NOI1995)题解
题目描述
在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分。
试设计出1个算法,计算出将N堆石子合并成1堆的最小得分和最大得分.
输入输出格式
输入格式:
数据的第1行试正整数N,1≤N≤100,表示有N堆石子.第2行有N个数,分别表示每堆石子的个数.
输出格式:
输出共2行,第1行为最小得分,第2行为最大得分.
1995的NOI题目,然而却是一道非常水的区间DP。
区间DP,顾名思义,求区间最值问题。通过小区间来更新大区间,最后逐渐更新出答案。
区间DP常用枚举套路:
- for(int len = ;len<=n;len++)
- {
- for(int i = ;i+len-<=n;i++)
- {
- int j = i+len-;
- dp[i][j] = ...
- }
- }
外层枚举长度,下一层枚举初始端点,终点通过长度+起点-1枚举出来,需要注意的是起点枚举范围是i+len-1,也就是终点要在区间长度以内。
继续说这道题目。
大区间一定是通过小区间合并出来的,这也是我们使用区间DP的原因。但是这道题并不是一条链上的石子,而是一个环。也就是说我们有可能在最后一个石子回头,与前面的石子合并。
那我们只需要把原来的链的长度变成二倍(除了最后一位),而枚举长度仍然是一倍的长度不就好了?
举个例子:
2,3,4,5
我们可以把它变成2,3,4,5,2,3,4
枚举的时候仍然是4的长度。这样就完美的处理了链的情况。
接下来是状态转移方程。
不知道有没有同学会和我开始时候有一样的错觉,全部合并到一起不就是所有的值相加吗?
然而并不是这样的,当一个区间与另一个区间合并时,原来的区间的数被算了两次。
再举个栗子。
[1,2] 与[2,3]合并
前者合并之后是3,后者是5
这样在合并一次就是3+5+8
相当于1+2+2+3+1+2+2+3
虽然这个结论是错的,但是我们从中可以得到一个结论,每次合并的时候,区间内的所有值都会被再次算一遍。
所以要预处理前缀和。
接下来说方程。
dp[i][j]表示把[i][j]中的石子合并成一堆所需要的费用。
[i][j]之间我们可以选择任意一个点,把这个区间分成两段,通过这两段合并成这一段区间。
所以
dpmax[i][j] = max(dpmax[i][j],dpmax[i][k-]+dpmax[k][j]+before[j]-before[i-]);
dpmin[i][j] = min(dpmin[i][j],dpmin[i][k-]+dpmin[k][j]+before[j]-before[i-]);
before代表前缀和
每个点的初始值就是它自己的花费。
最后上代码。
- #include<cstdio>
- #include<algorithm>
- #include<cstring>
- #include<string>
- using namespace std;
- int num[];
- int before[];
- int dpmax[][];
- int dpmin[][];
- int main()
- {
- int n;
- scanf("%d",&n);
- for(int i = ;i<=n;i++)
- {
- scanf("%d",&num[i]);
- num[n+i] = num[i];
- before[i] = before[i-]+num[i];
- }
- for(int i = n+;i<=*n-;i++)
- {
- before[i] = before[i-]+num[i];
- }
- memset(dpmin,0x3f,sizeof(dpmin));
- memset(dpmax,-,sizeof(dpmax));
- for(int i = ;i<=*n-;i++)
- {
- dpmin[i][i] = ;
- dpmax[i][i] = ;
- }
- for(int len = ;len<=n;len++)
- {
- for(int i = ;i+len-<=*n-;i++)
- {
- int j = i+len-;
- for(int k = i+;k<=j;k++)
- {
- dpmax[i][j] = max(dpmax[i][j],dpmax[i][k-]+dpmax[k][j]+before[j]-before[i-]);
- dpmin[i][j] = min(dpmin[i][j],dpmin[i][k-]+dpmin[k][j]+before[j]-before[i-]);
- }
- }
- }
- int ma = -;
- int mi = ;
- for(int i = ;i<=n;i++)
- {
- ma = max(ma,dpmax[i][i+n-]);
- mi = min(mi,dpmin[i][i+n-]);
- }
- printf("%d\n%d",mi,ma);
- return ;
- }
- /*
- 3
- 1 2 3
- */
石子合并(NOI1995)题解的更多相关文章
- [NOI1995]石子合并 题解
一道经典的dp题 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1个算法,计算出将N堆石子 ...
- 洛谷 P1880 [NOI1995]石子合并 题解
P1880 [NOI1995]石子合并 题目描述 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试 ...
- 石子合并(NOI1995)
石子合并(NOI1995) 时间限制: 1 Sec 内存限制: 128 MB提交: 90 解决: 48[提交][状态][讨论版] 题目描述 在操场上沿一直线排列着 n堆石子.现要将石子有次序地合并 ...
- 洛谷 P1880 [NOI1995] 石子合并(区间DP)
传送门 https://www.cnblogs.com/violet-acmer/p/9852294.html 题解: 这道题是石子合并问题稍微升级版 这道题和经典石子合并问题的不同在于,经典的石子合 ...
- P1880 [NOI1995]石子合并[区间dp+四边形不等式优化]
P1880 [NOI1995]石子合并 丢个地址就跑(关于四边形不等式复杂度是n方的证明) 嗯所以这题利用决策的单调性来减少k断点的枚举次数.具体看lyd书.这部分很生疏,但是我还是选择先不管了. # ...
- 区间DP小结 及例题分析:P1880 [NOI1995]石子合并,P1063 能量项链
区间类动态规划 一.基本概念 区间类动态规划是线性动态规划的拓展,它在分阶段划分问题时,与阶段中元素出现的顺序和由前一阶段的那些元素合并而来由很大的关系.例如状态f [ i ][ j ],它表示以已合 ...
- P1880 [NOI1995]石子合并 区间dp
P1880 [NOI1995]石子合并 #include <bits/stdc++.h> using namespace std; ; const int inf = 0x3f3f3f3f ...
- 【区间dp】- P1880 [NOI1995] 石子合并
记录一下第一道ac的区间dp 题目:P1880 [NOI1995] 石子合并 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 代码: #include <iostream> ...
- [洛谷P1880][NOI1995]石子合并
区间DP模板题 区间DP模板Code: ;len<=n;len++) { ;i<=*n-;i++) //区间左端点 { ; //区间右端点 for(int k=i;k<j;k++) ...
随机推荐
- 解决本地Bootstrap字体图标不可见的问题
原文:https://www.jianshu.com/p/70ac459d33e7 作为Bootstrap的初学者,我最近遇到了一个问题:在使用Bootstrap字体图标时,图标不可见.使用代码如下: ...
- 用cProfile做性能分析【转】
原文地址: https://www.cnblogs.com/kaituorensheng/p/4453953.html
- js基础闭包练习题
题目描述 实现函数 makeClosures,调用之后满足如下条件:1.返回一个函数数组 result,长度与 arr 相同2.运行 result 中第 i 个函数,即 result[i](),结果与 ...
- js utc转当地时间
javascript utc转当地时间 后台传过来的时间:2019-07-03T01:39:51.691242+08:00 转成当地时间:2019-07-02 17:39:51 new Date(20 ...
- MapReduce1.x与MapReduce2.x差异
一.MapReduce1.x简介 1.图解 2.JobTracker 主节点,单点,负责调度所有的作用和监控整个集群的资源负载. 3.TaskTracker 从节点,自身节点资源管理和JobTrack ...
- linux 中常遇到的问题
1.上传文件是速度为零 xshell连接对应的Ubuntu服务器上在Ubuntu服务器上安装lrzszsudo apt install lrzsz xshell连接对应的centos服务器上 yum ...
- idea 启动项目报错,more than one fragment with the name [spring web] was found
这是由于idea导入项目的时候有多个模块,并且有多个web.xml导致的,先删除对应的模块,后启动即可. 另外也有可能是spring的jar冲突,把冲突的jar删除即可.
- githe和github连接,上传
Git入门 如果你完全没有接触过Git,你现在只需要理解通过Git的语法(敲入一些命令)就可以将代码上传到远程的仓库或者下载到本地的仓库(服务器),可知我们此时应该有两个仓库,就是两个放代码的地方,一 ...
- 使用10046追踪执行计划demo
(一)开启10046追踪 SQL> alter session set events '10046 trace name context forever,level 12'; (二)执行sql语 ...
- win7安装小米8驱动
首先尝试了小米助手,安装后提示说要会弹窗然后选择确定,但是并没有弹窗:尝试管理员方式运行,依然没有弹框.差评. 正确的姿势是:http://bigota.d.miui.com/tools/xiaomi ...