洛谷1880 石子合并

P1880 石子合并

题目描述

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

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

输入输出格式

输入格式:

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

输出格式:

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

输入输出样例

输入样例#1:

4
4 5 9 4

输出样例#1:

43
54

上午做题的时候,一个小同学问我此题,说实话我并没有做过环形的石子合并,这也算是练了练手

np[i][j]代表从i开始长度为j的一段石子合并起来的最小成绩

xp[i][j] 代表从i开始长度为j的一段石子合并起来的最大成绩

关于环形怎么解决,请看图

对于n种石子的情况,就会有n种计算方法,在这n个方法中取个最大或最小即可。

(蓝色石子是复制品)

代码这次写的比较工整

#include<iostream>
using namespace std;
int n,st[210],sum[210][210];
int xp[210][210],np[210][210],mn,mx;
int main(){
    cin>>n;
    for(int i=1;i<=n;i++)
    {cin>>st[i];st[i+n]=st[i];}
    for(int i=1;i<=n*2-1;i++)sum[1][i]=st[i]+sum[1][i-1];
    for(int i=1;i<=n*2-1;i++)
        for(int j=i;j<=n*2-1;j++)
            sum[i][j]=sum[1][j]-sum[1][i-1];
    for(int i=n*2;i>=1;i--){
        for(int j=i+1;j<=min(i+n,n*2);j++){
            np[i][j]=99999999;
            for(int k=i;k<j;k++){
                xp[i][j]=max(xp[i][k]+xp[k+1][j]+sum[i][k]+sum[k+1][j],xp[i][j]);
                np[i][j]=min(np[i][k]+np[k+1][j]+sum[i][k]+sum[k+1][j],np[i][j]);
            }
        }
    }
    mn=99999999;
    for(int i=1;i<=n-1;i++){
            mn=min(mn,np[i][i+n-1]);
            mx=max(mx,xp[i][i+n-1]);
        }
    cout<<mn<<endl<<mx;
}
时间: 2024-12-20 07:41:13

洛谷1880 石子合并的相关文章

codevs 1048/洛谷 1880:石子归并

题目描述 Description 有n堆石子排成一列,每堆石子有一个重量w[i], 每次合并可以合并相邻的两堆石子,一次合并的代价为两堆石子的重量和w[i]+w[i+1].问安排怎样的合并顺序,能够使得总合并代价达到最小. 输入描述 Input Description 第一行一个整数n(n<=100) 第二行n个整数w1,w2...wn  (wi <= 100) 输出描述 Output Description 一个整数表示最小合并代价 样例输入 Sample Input 4 4 1 1 4 样

洛谷 P1880 石子合并

题目描述 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1个算法,计算出将N堆石子合并成1堆的最小得分和最大得分. 输入输出格式 输入格式: 数据的第1行试正整数N,1≤N≤100,表示有N堆石子.第2行有N个数,分别表示每堆石子的个数. 输出格式: 输出共2行,第1行为最小得分,第2行为最大得分. 输入输出样例 输入样例#1: 4 4 5 9 4 输出样例#1: 43 54 区间dp

[codevs1048]石子归并&amp;&amp;[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 s

[洛谷P1631]序列合并

题目大意:给你两个非降序序列a和b,每个序列n个数,在a和b各取一个数相加,能得到$n^2$个和,求这些和中最小的n个. 解题思路:我们先把a[1]+b[1],a[1]+b[2],a[1]+b[3]...a[1]+b[n]扔进一个小根堆里,每次取出一个数,设为a[i]+b[j],那么我们输出这个数,并把a[i+1]+b[j]放进堆里,循环n次就能得出答案.时间复杂度$O(n\log n)$. 我用pbds的优先队列+pair实现,用第二个数保存a[i]+b[j]的i值. 然而我第一次push时i

【洛谷】P1090 合并果子

这道题很经典 . 考点:二叉堆,贪心 我写了两个版本.顺便复习一下手工堆. 系统堆版: #include<bits/stdc++.h> using namespace std; priority_queue <int> q; int main() { int n; long long ans = 0; scanf("%d",&n); for (int i = 1 , x ; i <= n ; i ++) { scanf("%d"

动态规划—石子合并(直线和环)

先来看直线的: N堆石子摆成一条线.现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆石子合并成新的一堆,并将新的一堆石子数记为该次合并的代价.计算将N堆石子合并成一堆的最小代价. 例如: 1 2 3 4,有不少合并方法 1 2 3 4 => 3 3 4(3) => 6 4(9) => 10(19) 1 2 3 4 => 1 5 4(5) => 1 9(14) => 10(24) 1 2 3 4 => 1 2 7(7) => 3 7(10) =>

[题解]洛谷比赛『期末考后的休闲比赛2』

[前言] 这场比赛已经结束了有几天,但我各种忙,虽然AK但还是没来得及写题解.(我才不会告诉你我跑去学数据结构了) T1 区间方差 (就不贴题好了) 首先可以推公式(我们可以知道,线段树然而并不能通过初中学过的方差公式在log(L)内求出方差): (s2表示方差,L表示区间长度,xi表示区间的每一项,最后一个x上画了一根线表示这些数据的平均数) 用二项式定理完全平方公式可得: 再次展开: 另外,再代入以下这个 得到了: 然后继续吧.. 然后duang地一声合并同类项,于是我们得到了: 然后可以高

洛谷P2420 让我们异或吧

P2420 让我们异或吧 161通过 450提交 题目提供者该用户不存在 标签洛谷原创云端↑ 难度普及/提高- 时空限制1s / 128MB 提交  讨论  题解 最新讨论更多讨论 倍增可做的吧 玄学 更改根节点得分不一样- 这题面似乎对一些群体不太友- 这题为什么没数据 C++选手注意了 题目描述 异或是一种神奇的运算,大部分人把它总结成不进位加法. 在生活中-xor运算也很常见.比如,对于一个问题的回答,是为1,否为0.那么: (A是否是男生 )xor( B是否是男生)=A和B是否能够成为情

洛谷P1848 [USACO12OPEN]书架Bookshelf

当农夫约翰闲的没事干的时候,他喜欢坐下来看书.多年过去,他已经收集了 N 本书 (1 <= N <= 100,000), 他想造一个新的书架来装所有书. 每本书 i 都有宽度 W(i) 和高度 H(i).书需要按顺序添加到一组书架上:比如说,第一层架子应该包含书籍1 ... k,第二层架子应该以第k + 1本书开始,以下如此.每层架子的总宽度最大为L(1≤L≤1,000,000,000).每层的高度等于该层上最高的书的高度,并且整个书架的高度是所有层的高度的总和,因为它们都垂直堆叠. 请帮助农