一、题目描述
合并果子(fruit)
题目描述
在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆。 多多决定把所有的果子合成一堆。每一次合并,多多可以把两堆果子合并到一起,消耗的体力等于两堆果子的重量之和。可以看出,所有的果子经过n-1次合并之后,就只剩下一堆了。多多在合并果子时总共消耗的体力等于每次合并所耗体力之和。因为还要花大力气把这些果子搬回家,所以多多在合并果子时要尽可能地节省体力。 假定每个果子重量都为1,并且已知果子的种类数和每种果子的数目,你的任务是设计出合并的次序方案,使多多耗费的体力最少,并输出这个最小的体力耗费值。例如有3种果子,数目依次为1,2,9。可以先将 1、2堆合并,新堆数目为3,耗费体力为3。接着,将新堆与原先的第三堆合并,又得到新的堆,数目为12,耗费体力为 12。所以多多总共耗费体力=3+12=15。可以证明15为最小的体力耗费值。
输入
第1行:一个整数n(1≤n≤10000),表示果子的种类数。第2行:包含n个整数,用空格分隔,第i个整数ai(1≤ai≤20000)是第i种果子的数目。
输出
一行只包含一个整数,也就是最小的体力耗费值。输入数据保证这个值小于2^31。
样例输入
3
1 2 9
样例输出
15
二、分析
这一道题有点像动态规划的“合并石子”,但是合并石子只能合并相邻的两堆,这一道题可以合并任意两堆。
从总体思路来讲,我们应该先合并数量最小的两堆,就要先进行一次排序(建议用:堆排序,归并排序和快速排序)
就题目来说,尽量用堆排序,因为每次果子合并之后,又要把合并后的数据重新进行排序。
所以这一道题用堆排序就可以快速处理新数据。
法1:在原数组中直接进行合并,合并后就把后面的数据向前移动,移动了之后把长度减1,再进行快速排序。
但这种算法果断超时,因为数据的大小在10000以内,大量地移动数据和多次进行排序,就消耗了许多时间。
以下的代码超时了7个点:
#include
#include
#include
using namespace std;
int a[10005],q,n,b[10005];
void put(int k)
{
a[q++]=k;
//push_heap(a,a+q);
push_heap(a,a+q,greater());
}
int get()
{
//pop_heap(a,a+w);
pop_heap(a,a+q,greater());
return a[--q];
}
int main()
{
//freopen("fruit.in","r",stdin);
//freopen("fruit.out","w",stdout);
int i,j,x,n,sum=0;
scanf("%d",&n);
for(i=0;i1;){
b[0]=b[0]+b[1];
sum+=b[0];
for(j=1;j
法2:用堆排序之后,把前面的两个数get()出来,再加起来,把两个数的和put()进去,进行一次堆的维护。然后再把
前两个数get()出来...(重复以上操作),该算法比较简单,复杂度较低,速度快。(建议用此方法)
#include
#include
#include
using namespace std;
int a[10005],q,n,b[10005];
void put(int k)
{
a[q++]=k;
//push_heap(a,a+q);
push_heap(a,a+q,greater());
}
int get()
{
//pop_heap(a,a+w);
pop_heap(a,a+q,greater());
return a[--q];
}
int main()
{
//freopen("fruit.in","r",stdin);
//freopen("fruit.out","w",stdout);
int i,x,n,sum=0;
scanf("%d",&n);
for(i=0;i
法3:运用优先队列中优先的顺序来进行排序,做法和上面的方法差不多,但是速度比上面的方法慢一些,
因为优先队列要保证所有的数据都按照优先的顺序排列,而堆只需要保证顶层数据(即第一个数据)最优就可以了。
#include
#include
#include
using namespace std;
priority_queue,greater >h;
int main()
{
freopen("fruit2.in","r",stdin);
freopen("fruit2.out","w",stdout);
int i,x,n,y,sum=0;
scanf("%d",&n);
for(i=0;i