石子合并(3种变形)

<1>

题目:

N堆石子排成一排(n<=100),现要将石子有次序地合并成一堆,规定每次只能选相邻的两堆合并成一堆,并将新的一堆的石子数,记为改次合并的得分,编一程序,由文件读入堆数n及每堆石子数(<=200);

(1)选择一种合并石子的方案,使得做n-1次合并,得分的总和最少;

(2)选择一种合并石子的方案,使得做n-1次合并,得分的总和最多;

输入格式

第一行为石子堆数n

第二行为每堆石子数,每两个数之间用一空格分隔。

输出格式

从第1行为得分最小

2行是得分最大。

样例

样例输入

 4
 4 5 9 4

样例输出

 44
 54

特点:

仅是一排石子,而且只能是相邻两个石子相邻可以合并,所以第1个石子只能与右边的合并,第n个石子只能与左边的合并,所以用区间dp简单枚举合并即可。

数组表示:(以下同)

f[i][j]:从ij的区间的最大/最小值;

sum[i]i的前缀和(便于计算);

动态规划:

阶段:

区间长度dd=1时初始值即石子本身分数,所以从d=2开始枚举,以此到d=n

状态:

ij的区间的最大/最小值;

决策:

是否将从ik的区间与从k+1j的区间合并,若合并即加上sum[i]-sum[j-1]ij的分数);

动态转移方程:

f[i][j]=min(f[i][j],f[i][k]+f[k+1][j]+sum[i]-sum[j-1]);

f[i][j]=max(f[i][j],f[i][k]+f[k+1][j]+sum[i]-sum[j-1]);

注意:

最小值f[i][j]的初始值为0x7f7f7f7f

最大值f[i][j]的初始值为0

代码:

 #include <bits/stdc++.h>
using namespace std;
const int maxn=+;
const int INF=0x3f3f3f3f;
int n;
int f1[maxn][maxn],f2[maxn][maxn];
int sum[maxn];
int a[maxn];
int minx=INF,maxx=;
void MIN(){
for(int d=;d<=n;d++){
for(int i=,j;(j=i+d-)<=n;i++){
for(int k=i;k<j;k++){
f1[i][j]=min(f1[i][j],f1[i][k]+f1[k+][j]+sum[j]-sum[i-]);
}
}
}
}
void MAX(){
for(int d=;d<=n;d++){
for(int i=,j;(j=i+d-)<=n;i++){
for(int k=i;k<j;k++){
f2[i][j]=max(f2[i][j],f2[i][k]+f2[k+][j]+sum[j]-sum[i-]);
}
}
}
}
int main(){
memset(f1,INF,sizeof(f1));
scanf("%d",&n);
for(int i=;i<=n;i++){
scanf("%d",&a[i]);
}
for(int i=;i<=n;i++){
if(i==){
sum[i]=a[i];
}else{
for(int j=;j<=i;j++){
sum[i]+=a[j];
}
}
}
for(int i=;i<=n;i++){
f1[i][i]=;
}
MIN();
MAX();
printf("%d\n",f1[][n]);
printf("%d\n",f2[][n]);
return ;
}

<2>

题目:

在一个园形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分。

试设计出1个算法,计算出将N堆石子合并成1堆的最小得分和最大得分。

输入格式

数据的第1行试正整数N,(1≤ N ≤100),表示有N堆石子。

2行有N个数,分别表示每堆石子的个数。

输出格式

输出共2行,第1行为最小得分,第2行为最大得分。

样例

样例输入

 4
 4 4 5 9

样例输出

 43
 54

特点:

与上一题不同的是,这里是圆形操场,即现在的石子是一个环,现在第1个石子与第n个石子算是相邻,也是很简单,自己将数组存储两边石子的分数,便可以模拟环,然后直接按上次的dp处理即可。

注意:

此时的d还是枚举到n

ij则需要枚举到2*n

代码:

 #include <bits/stdc++.h>
using namespace std;
const int maxn=+;
const int INF=0x3f3f3f3f;
int n;
int f1[maxn][maxn],f2[maxn][maxn];
int sum[maxn];
int a[maxn];
int minx=INF,maxx=;
void MIN(){
for(int d=;d<=n;d++){
for(int i=,j;(j=i+d-)<=*n;i++){
f1[i][j]=INF;
for(int k=i;k<j;k++){
f1[i][j]=min(f1[i][j],f1[i][k]+f1[k+][j]+sum[j]-sum[i-]);
}
}
}
}
void MAX(){
for(int d=;d<=n;d++){
for(int i=,j;(j=i+d-)<=*n;i++){
f2[i][j]=;
for(int k=i;k<j;k++){
f2[i][j]=max(f2[i][j],f2[i][k]+f2[k+][j]+sum[j]-sum[i-]);
}
}
}
}
int main(){
scanf("%d",&n);
for(int i=;i<=n;i++){
scanf("%d",&a[i]);
a[n+i]=a[i];
}
for(int i=;i<=*n;i++){
sum[i]=sum[i-]+a[i];
}
MIN();
MAX();
int zuix=INF,zuid=;
for(int i=;i<=n;i++){
zuix=min(zuix,f1[i][i+n-]);
zuid=max(zuid,f2[i][i+n-]);
}
printf("%d\n%d\n",zuix,zuid);
return ;
}

<3>

题目:

在一个园形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分。

试设计出1个算法,计算出将N堆石子合并成1堆最大得分。

输入格式

数据的第1行试正整数N,1≤N≤2000,表示有N堆石子.第2行有N个数,分别表示每堆石子的个数。

输出格式

输出共1行,最大得分。

样例

样例输入

 4
 4 4 5 9

样例输出

 54

注意:

虽然一看这题与<2>完全一样,但是仔细看的话,N的范围发生了变化,那么就不能与<1>、<2>一样用直接dp的方法,上两次只有N3的​效率,所以这次得需要优化。

区间dp的优化方式有四边不等式法,但是这里是求最大值,没有单调性,无法用四边不等式。

但是最大值有一个性质,总是在两个端点的最大者中取到。

动态转移方程式:

f[i][j]=max(f[i][j-1],f[i+1][j])+sum[j]-sum[i-1]

代码:

 
#include<bits/stdc++.h>
using namespace std;
const int maxn=1e6+,INF=0x3f3f3f3f;
int n,m,f[][],ans,a[maxn],sum[];
int main(){
cin>>n;
for(int i=;i<=n;i++){
cin>>a[i];
a[i+n]=a[i];
sum[i]=a[i]+sum[i-];
}
for(int i=n+;i<=*n;i++){
sum[i]=sum[i-]+a[i];
}
for(int d=;d<=n;d++){
for(int i=,j;(j=i+d-)<=n*;i++){
f[i][j]=max(f[i][j-],f[i+][j])+sum[j]-sum[i-];
}
}
int ansmax=;
for(int i=;i<=n;i++){
ansmax=max(ansmax,f[i][i+n-]);
}
cout<<ansmax<<endl;
return ;
}

石子合并——区间dp的更多相关文章

  1. 洛谷 P1880 [NOI1995] 石子合并(区间DP)

    传送门 https://www.cnblogs.com/violet-acmer/p/9852294.html 题解: 这道题是石子合并问题稍微升级版 这道题和经典石子合并问题的不同在于,经典的石子合 ...

  2. 石子合并 区间dp模板

    题意:中文题 Description 在操场上沿一直线排列着 n堆石子.现要将石子有次序地合并成一堆.规定每次只能选相邻的两堆石子合并成新的一堆, 并将新的一堆石子数记为该次合并的得分.允许在第一次合 ...

  3. HDU4632 Poj2955 括号匹配 整数划分 P1880 [NOI1995]石子合并 区间DP总结

    题意:给定一个字符串 输出回文子序列的个数    一个字符也算一个回文 很明显的区间dp  就是要往区间小的压缩! #include<bits/stdc++.h> using namesp ...

  4. 石子合并 区间DP模板题

    题目链接:https://vjudge.net/problem/51Nod-1021 题意 N堆石子摆成一条线.现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆石子合并成新的一堆,并将新的一堆石 ...

  5. 洛谷 P1080 石子合并 ( 区间DP )

    题意 : 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分.试设计出1个算法,计算出将N堆石子合并成1堆 ...

  6. 洛谷P1880 石子合并(环形石子合并 区间DP)

    题目描述 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1个算法,计算出将N堆石子合并成1 ...

  7. HRBUST - 1818 石子合并 区间dp入门

    有点理解了进阶指南上说的”阶段,状态和决策“ /* 区间dp的基础题: 以区间长度[2,n]为阶段,枚举该长度的区间,状态dp[l][r]表示合并区间[l,r]的最小费用 状态转移方程dp[l][r] ...

  8. HDU 3506 (环形石子合并)区间dp+四边形优化

    Monkey Party Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 131072/65536 K (Java/Others)Tot ...

  9. P1880 [NOI1995]石子合并 区间dp+拆环成链

    思路 :一道经典的区间dp  唯一不同的时候 终点和起点相连  所以要拆环成链  只需要把1-n的数组在n+1-2*n复制一遍就行了 #include<bits/stdc++.h> usi ...

随机推荐

  1. 用vue实现一个简单的时间屏幕

    前言 去年用了一个小的 app,叫做 一个木函,本来想着用来做动画优化就删掉了的,不过看到他有个时间屏幕的小工具,就点进去看了下,觉得挺好玩的,就想着能不能自己实现一下. ps: 闲话不多说,先上例子 ...

  2. 循序渐进VUE+Element 前端应用开发(7)--- 介绍一些常规的JS处理函数

    在我们使用VUE+Element 处理界面的时候,往往碰到需要利用JS集合处理的各种方法,如Filter.Map.reduce等方法,也可以设计到一些对象属性赋值等常规的处理或者递归的处理方法,以前对 ...

  3. Netty源码学习系列之4-ServerBootstrap的bind方法

    前言 今天研究ServerBootstrap的bind方法,该方法可以说是netty的重中之重.核心中的核心.前两节的NioEventLoopGroup和ServerBootstrap的初始化就是为b ...

  4. test for OCr

  5. Ubuntu18.04下MySQL8.0和Navicat15的安装与使用

    目录 一.MySQL8.0安装 二.Navicat安装并与MySQL连接 一.MySQL8.0安装 注意:若直接 sudo apt install mysql-server,你会发现安装后的版本是5. ...

  6. 关于JSON数据体积优化的一点小心得

    最近在做的一个项目里传输的json数据比较大,造成了线程间的卡顿,于是想优化一下json数据的体积. 可以看到在json文件里有很多无用的字段,这些字段占据了大量的存储空间. 对数据的结构作一下优化, ...

  7. excel 如何制作带下拉框的动态折线图表

    首先我们需要有个类似下图产品销量的基础数据表. 首先将光标放入表格中任意位置,然后插入一个不带点标记的折线图,然后将折线的颜色设置为灰色. 第一次设置成灰色后,一定善用f4快捷键进行快速的折线颜色设置 ...

  8. python中turtle模块画正多边形

    画正多边形主要是计算多边形每个角度对应的外角的度数,计算出来这个度数即可画图,相对来说非常简单 以正六边形为例 import turtle import time t = turtle.Pen() f ...

  9. CKA考试个人心得分享

    考试相关准备: 真题:需要的私密: 网络:必须开启VPN,以便能访问国外网络,强烈建议在香港搭建相应FQ: 证件:考试需要出示含有拉丁文(英文)带照片的有效证件,相关有效证件参考(优先建议护照):ht ...

  10. C#构造函数 -0028

    默认构造函数 声明基本构造函数的语法就是声明一个与类同名的方法,但该方法没有返回类型: public class MyClass { public MyClass() { } // rest of c ...