[codevs1048]石子归并&[codevs2102][洛谷P1880]石子归并加强版
codevs1048:
题目大意:有n堆石子排成一列,每次可合并相邻两堆,代价为两堆的重量之和,求把他们合并成一堆的最小代价。
解题思路:经典区间dp。设$f[i][j]$表示合并i~j的石子需要的最小代价。则有$f[i][j]=min(f[i][k]+f[k+1][j]+\sum\limits _{l=i}^{j}a[l])$,时间复杂度$O(n^3)$。
C++ Code:
#include<cstdio>
#include<cstring>
using namespace std;
int n,a[102],f[102][102],s[102];
int main(){
scanf("%d",&n);
memset(f,0x3f,sizeof f);
for(int i=1;i<=n;++i)scanf("%d",&a[i]),s[i]=s[i-1]+a[i],f[i][i]=0;
for(int i=n;i;--i)
for(int j=i+1;j<=n;++j)
for(int k=i;k<j;++k)
if(f[i][j]>f[i][k]+f[k+1][j]+s[j]-s[i-1])f[i][j]=f[i][k]+f[k+1][j]+s[j]-s[i-1];
printf("%d\n",f[1][n]);
return 0;
}
注意代码第9行,为什么i要倒着循环?举个栗子,如果要求f[1][10],就有f[1][10]=min(f[1][10],f[1][5]+f[6][10]+sum[1][10]),但是i才循环到1,就需要f[6][10]的结果,于是导致答案错误。而倒着循环,就可保证i+1~n的所有数据都已求完,就不会导致答案错误了。
codevs2102&&洛谷P1880:
题目大意:有n堆石子摆成环状,每次可合并相邻两堆,代价为两堆的重量之和,求把他们合并成一堆的最小代价和最大代价。
解题思路:本题除了是个环以外,和上题没什么区别。我们可以用化环为链的方法,具体的实现就是将这个环的单圈复制一遍,然后做n次上述dp即可。求最大价值就是把状态转移方程里的$min$改成$max$即可。时间复杂度$O(n^4)$,但代码运行量应该是不到这个极限的。
C++ Code:
#include<cstdio>
#include<cstring>
using namespace std;
int n,a[202],fmax[202][202],fmin[202][202],s[202],Max=0,Min=200000000;
void dp(int h){
for(int i=n;i;--i){
fmax[i+h][i+h]=fmin[i+h][i+h]=0;
for(int j=i+1;j<=n;++j){
fmax[i+h][j+h]=0;
fmin[i+h][j+h]=200000000;
for(int k=i;k<j;++k){
if(fmax[i+h][j+h]<fmax[i+h][k+h]+fmax[k+h+1][j+h]+s[j+h]-s[i+h-1])
fmax[i+h][j+h]=fmax[i+h][k+h]+fmax[k+h+1][j+h]+s[j+h]-s[i+h-1];
if(fmin[i+h][j+h]>fmin[i+h][k+h]+fmin[k+h+1][j+h]+s[j+h]-s[i+h-1])
fmin[i+h][j+h]=fmin[i+h][k+h]+fmin[k+h+1][j+h]+s[j+h]-s[i+h-1];
}
}
}
if(Max<fmax[1+h][n+h])Max=fmax[1+h][n+h];
if(Min>fmin[1+h][n+h])Min=fmin[1+h][n+h];
}
int main(){
scanf("%d",&n);
for(int i=1;i<=n;++i)scanf("%d",&a[i]),s[i]=s[i-1]+a[i];
for(int i=n+1;i<2*n;++i)
a[i]=a[i-n],s[i]=s[i-1]+a[i];
for(int i=1;i<=n;++i)
dp(i-1);
printf("%d\n%d\n",Min,Max);
return 0;
}
[codevs1048]石子归并&[codevs2102][洛谷P1880]石子归并加强版的更多相关文章
- 洛谷P1880 石子合并(区间DP)(环形DP)
To 洛谷.1880 石子合并 题目描述 在一个园形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1 ...
- 经典DP 洛谷p1880 石子合并
https://www.luogu.org/problemnew/show/P1880 题目 题目描述 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新 ...
- 洛谷P1880 石子合并(环形石子合并 区间DP)
题目描述 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1个算法,计算出将N堆石子合并成1 ...
- 洛谷 P1880 石子合并
题目描述 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1个算法,计算出将N堆石子合并成1 ...
- 洛谷P1880 石子合并
经典水题....... 断环为链长度乘二,求前缀和区间DP. #include <cstdio> #include <cstring> #include <algorit ...
- 洛谷P1880 [NOI1995]石子合并 纪中21日c组T4 2119. 【2016-12-30普及组模拟】环状石子归并
洛谷P1880 石子合并 纪中2119. 环状石子归并 洛谷传送门 题目描述1 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石 ...
- codevs 1048/洛谷 1880:石子归并
题目描述 Description 有n堆石子排成一列,每堆石子有一个重量w[i], 每次合并可以合并相邻的两堆石子,一次合并的代价为两堆石子的重量和w[i]+w[i+1].问安排怎样的合并顺序,能够使 ...
- 洛谷 P1880 [NOI1995] 石子合并(区间DP)
传送门 https://www.cnblogs.com/violet-acmer/p/9852294.html 题解: 这道题是石子合并问题稍微升级版 这道题和经典石子合并问题的不同在于,经典的石子合 ...
- 洛谷 P6031 - CF1278F Cards 加强版(推式子+递推)
洛谷题面传送门 u1s1 这个推式子其实挺套路的吧,可惜有一步没推出来看了题解 \[\begin{aligned} res&=\sum\limits_{i=0}^ni^k\dbinom{n}{ ...
随机推荐
- 速学JavaScript!
什么是JavaScript? JavaScript是一种轻量级的脚本语言,也是一种嵌入式语言,是一种对象模型语言,简称JS:JavaScript的核心语法部分(语言本身)很精简,只包括两个部分: 基本 ...
- [Codeforces 115E]Linear Kingdom Races
题目大意: 有n块地,初始是荒地.你可以把某些荒地开垦(需要花费相应的价值\(a_i\)(正整数)),然后这些荒地就可以种田. 现在有m年,每年要在l到r区间内种田,获得p(正整数)的价值(必须保证l ...
- python dns 服务器
import socketserver import struct import threading # DNS Query class SinDNSQuery: def __init__(self, ...
- 归档 SCP SFTP RSYNC(数据同步)
tar 选项 目标文件 源文件(1 2 3) tar cf **.tar file1 file2 file3 (默认情况下 cf选项只有归档没有压缩) tar xf 从归档中提取 创建tar的存档 ...
- pytorch 3 activation 激活函数
2.3 Activation Function import torch import torch.nn.functional as F from torch.autograd import Vari ...
- [转]Python常用字符串
转自:http://blog.csdn.net/daemonpei/article/details/6325762 字符串相关操作: + :string1+string2 #联接字符串,将后一个串链接 ...
- sql学习笔记(18)-----------数据库创建过程
手动创建数据库的步骤: 第一步:决定数据库实例的SID 数据库实例的SID用来将当前实例和以后可能创建的实例进行区分 % setenv ORACLE_SID mynewdb 第二步:建立数 ...
- 华硕 X201E 拆机
每次笔记本拆机,装好之后.就会发现多了几个螺丝,忘了从哪拧下来了 以下记录下华硕 X201E 清灰拆机过程 1:电脑正面图 2:背面图,一共9个螺丝 3:背面的9个螺丝拧下来,把后盖沿着缝隙扣下来 w ...
- C++11新特性应用--介绍几个新增的便利算法(用于分区的几个算法)
今天继续. C++11新增的关于Non-modifying sequence operations和Modifying sequence operations的算法已经写了.具体信息见之前的博客. 以 ...
- Hadoop自学笔记(一)常见Hadoop相关项目一览
本自学笔记来自于Yutube上的视频Hadoop系列.网址: https://www.youtube.com/watch?v=-TaAVaAwZTs(当中一个) 以后不再赘述 自学笔记,难免有各类错误 ...