Time Limit: 1 second

Memory Limit: 50 MB

【问题描述】

在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆。多多决定把所有的果子合成一堆。 每一次合并,多多可以把两堆果子合并到一起,消耗的体力等于两堆果子的重量之和。可以看出,所有的果子经过n-1次合并之后,就只剩下一堆了。多多在合并果子时总共消耗的体力等于每次合并所耗体力之和。 因为还要花大力气把这些果子搬回家,所以多多在合并果子时要尽可能地节省体力。假定每个果子重量都为1,并且已知果子的种类数和每种果子的数目,你的任务是设计出合并的次序方案,使多多耗费的体力最少,并输出这个最小的体力耗费值。
例如有3种果子,数目依次为1,2,9。可以先将1、2堆合并,新堆数目为3,耗费体力为3。接着,将新堆与原先的第三堆合并,又得到新的堆,数目为12,耗费体力为12。所以多多总共耗费体力=3+12=15。可以证明15为最小的体力耗费值。

【输入】

共2行;

    第一行是一个整数n(1<=n<=10000),表示果子的种类数。

    第二行是第二行包含n个整数,用空格分隔,第i个整数ai(1<=ai<=20000)是第i种果子的数目。

【输出】

包含1行,这一行只包含一个整数,也就是最小的体力耗费值。输入数据保证这个值小于231。

【输入样例】

    3
    1 2 9

【输出样例】

    15

【 数据规模】

30%的数据满足:n<=1000

    50%的数据满足:n<=5000

100%的数据满足:n<=10000

【题解】

法一:将所有的果子数目快排一遍,然后合并前两个。再把合并的放到序列中(插入排序)

重复此步骤n-1次即可。

法二:用一个小根堆来维护整个序列的最小值。建堆的话,一开始要从倒数第二层开始建。然后往下调整。这样可以保证建堆是正确的。然后依然是重复n-1次操作。每次操作:先取出dui[1]即堆中的最小值。然后把堆尾的元素放到1位置。然后堆的大小递减。然后从1开始往下进行调整。维护堆。取出来的第一个数字为a,然后同样的,取出经过维护之后的堆中的第一个元素即最小值,设为b,然后堆的大小递减。同样从第一个元素开始往下调整堆。然后c=a+b,再把堆的大小递增,把c放到堆的最后面。然后从堆的最后一个位置往上进行调整。获取这个新插入的元素它最后的位置。然后从那个位置再往下进行调整即可。然后重复此步骤n-1次即可。

【代码】

#include <cstdio>

int n,dui[20001],size,pos,sum = 0;

void down_tiaozheng(int p) //往下调整
{
int x =dui[p]; //先记录要被调整的数字的大小。
int i = p,j = p*2; //p*2就表示其左儿子
while (j <= size) //如果没有超过树的范围
{
if (j < size && dui[j+1] < dui[j]) //获取它们之中较小的那个数。这样经过调整这个dui[j]
j++; //可能会往上交换。也能满足每个节点都比儿子节点小了。
if (x > dui[j]) //如果要进行交换 则交换
{
dui[i] = dui[j];
i = j;
j = i*2;
} //不用交换就可以结束了。
else
break;
}
dui[i] = x; //把x放到某个位置;
} void up_tiaozheng(int p) //往上调整
{
int x = dui[p]; //记录下要调整的数字大小
int i = p,j = p/2; //除2表示往上进行调整 即其父亲节点。
while (j > 0) //如果没有超过树的范围。则继续
{
if (dui[j] > x) //父节点只有一个。所以不用再判断了。
{
dui[i] = dui[j];//进行调整
i = j;
j = j/2;
}
else //如果不会出现冲突了
break; //就结束调整的过程。
}
dui[i] = x; //把它调整到i位置
pos = i; //然后记录它调整到了哪个位置。方便之后再往下调整。
} void input_data()
{
scanf("%d",&n);
for (int i = 1;i <= n;i++)
scanf("%d",&dui[i]);
size = n;
} void get_ans()
{
for (int i = (n/2);i>=1;i--) //输入数据完之后要进行建堆的操作,从倒数第二层开始调整以建堆
down_tiaozheng(i);
for (int i = 1;i <= n-1;i++)
{
int a = dui[1]; //取出两次堆中的第一个元素。然后调整两次堆。(取完就调整)
dui[1] = dui[size];
size--;
down_tiaozheng(1);
int b = dui[1];
dui[1] = dui[size];
size--;
down_tiaozheng(1);
int c = a+b;
sum+=c;
size++;
dui[size] = c; //把累加的c加到堆中
up_tiaozheng(size); //先往上调整。再往下调整。
down_tiaozheng(pos);
}
} void output_ans()
{
printf("%d\n",sum);
} int main()
{
//freopen("F:\\rush.txt","r",stdin);
input_data();
get_ans();
output_ans();
return 0;
}

【9602】&&【b402】合并果子的更多相关文章

  1. 【noip 2004】 合并果子

    noip2016结束后的第一份代码--优先队列的练习 合并果子 原题在这里 #include <iostream> #include <queue> #include < ...

  2. 合并果子 2004年NOIP全国联赛普及组

    时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description 在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆 ...

  3. NOIP2004合并果子

    题目描述 在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多决定把所有的果子合成一堆. 每一次合并,多多可以把两堆果子合并到一起,消耗的体力等于两堆果子的重量之和.可 ...

  4. codevs 1063 合并果子//优先队列

    1063 合并果子 2004年NOIP全国联赛普及组  时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石    题目描述 Description 在一个果园里,多多已经将所有的果 ...

  5. [KOJ6024]合并果子·改(强化版)

    [COJ6024]合并果子·改(强化版) 试题描述 在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多把这些果子堆排成一排,然后所有的果子合成一堆.    每一次合并 ...

  6. [KOJ6023]合并果子·改

    [COJ6023]合并果子·改 试题描述 在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多把这些果子堆排成一排,然后所有的果子合成一堆.    每一次合并,多多可以 ...

  7. [KOJ0574NOIP200406合并果子]

    [COJ0574NOIP200406合并果子] 试题描述 在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多决定把所有的果子合成一堆.    每一次合并,多多可以把两 ...

  8. NOIP提高组2004 合并果子题解

    NOIP提高组2004 合并果子题解 描述:在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多决定把所有的果子合成一堆. 每一次合并,多多可以把两堆果子合并到一起,消 ...

  9. 【NOIP合并果子】uva 10954 add all【贪心】——yhx

    Yup!! The problem name reects your task; just add a set of numbers. But you may feel yourselvesconde ...

随机推荐

  1. 1.15 Python基础知识 - 函数

    函数是可重用的程序代码段. 一.函数的声明和调用 声明格式: def 函数名([形参列表]): 函数体 调用格式: 函数名([实参列表]) 函数名:是一种标识符,命名规则为全小写字母,可以使用下划线增 ...

  2. 15、python学习手册之:列表和字典

    1.列表属于可变序列,支持在原处的修改 2.在标准python解锁器内部,列表就是C数组而不是链接结构 3.内置函数map对序列中的各项应用一个函数并把结果收集到一个新的列表中 eg:list(map ...

  3. 二、MongoDB基础知识

    1.文档是MongoDB的核心概念.文档就是键值对的一个有序集{'msg':'hello','foo':3}.类似于python中的有序字典. 需要注意的是: #1.文档中的键/值对是有序的. #2. ...

  4. [python]两种编程思维--面向过程和面向对象

    例如:eg:炒一份西红柿鸡蛋 一.面向过程 面向过程的编程思维,如下 二.面向对象 制作一台炒菜机器人,然后告诉机器人做一道西红柿炒鸡蛋.在这里,我们直接面对的是机器人,而非炒菜的过程,所以这里机器人 ...

  5. TPS54232-------电源管理芯片

    TPS54232 DC DC开关稳压器 电源管理芯片 放大器俗称功放 注意看芯片的次序1~8是如何排布的,这个规律一般是固定的 也许我们整理多了就能发现引脚的宽度和长度都是规格好的. 下面是封装: 所 ...

  6. 微服务实战(一):微服务架构的优势与不足 - DockOne.io

    原文:微服务实战(一):微服务架构的优势与不足 - DockOne.io [编者的话]本文来自Nginx官方博客,是微服务系列文章的第一篇,主要探讨了传统的单体式应用的不足,以及微服务架构的优势与挑战 ...

  7. GO语言学习(五)Go 语言基础语法

    Go 标记 Go 程序可以由多个标记组成,可以是关键字,标识符,常量,字符串,符号.如以下 GO 语句由 6 个标记组成: fmt.Println("Hello, World!") ...

  8. Mac怎么设置wifi热点

    苹果 Mac 系统中要把无线当作 Wifi 热点来用的话,需要电脑有其它网络接入才可以,也就是说它需要一个可以用于上网的网络,比如有线网络.尤其是对于使用 MacBook Pro 或 MacBook ...

  9. AE 向已存在的要素类中添加字段

    风过无痕 原文向已存在的要素类中添加字段 以前,在用AE写程序的时候,为了方便,一般都是直接新建一个MapControl窗体应用程序.这次需要解决的问题用不到窗口,就突发奇想,直接新建了一个Conso ...

  10. ImageView的圆角半径

    // 设置imageview的圆角半径 UIImageView *imageView = (UIImageView *)[cell viewWithTag:tag]; imageView.layer. ...