合并果子

首先来看一下题目:

(OI2004合并果子)

【题目描述】

果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆。多多决定把所有的果子合成一堆

每一次合并,多多可以把两堆果子合并到一起,消耗的体力等于两堆果子的重量之和。可以看出,所有的果子经过n-1次合并之后,就只剩下一堆了。多多在合并果子时总共消耗的体力等于每次合并所耗体力之和.

因为还要花大力气把这些果子搬回家,所以多多在合并果子时要尽可能地节省体力。假定每个果子重量都为1,并且已知果子的种类数和每种果子的数目,你的任务是设计出合并的次序方案,使多多耗费的体力最少,并输出这个最小的体力耗费值。

例如有3种果子,数目依次为1,2,9。可以先将1、2堆合并,新堆数目为3,耗费体力为3。接着,将新堆与原先的第三堆合并,又得到新的堆,数目为12,耗费体力为12。所以多多总共耗费体力=3+12=15。可以证明15为最小的体力耗费值。

【输入格式】

第一行:一个数n,表示果子的堆数。

第二行:n个数ai,表示每堆果子的果子个数。

【输出格式】

一个数:多多所要花费的最小的体力。

【数据范围】:

n≤100000

【输入样例1】

3

1 2 9

【输出样例1】

15

【输入样例2】

10

3 5 1 7 6 4 2 5 4 1

【输出样例2】

120

这个题就用到了堆的思想。按照一般的思想来打代码,应该是每一次合并之后就从大到小拍一遍序,接着合并。按照这样的思路,答案是没有错的,但是时间复杂度就会十分的高。下面附一下普通思路的合并果子的代码:

#include<iostream>
#include<cstdio>
#include<algorithm>
#define MAXN 100010
using namespace std;
},total=;
int cmp(int x,int y)
{
    return x>y;
}
int main()
{
    scanf("%d",&n);
    ;i<=n;i++)
     scanf("%d",&heavy[i]);
    sort(heavy+,heavy+n+,cmp);
    ;i<=n;i++)
    {
        heavy[i]+=heavy[i-];
        heavy[i-]=;
        sort(heavy+,heavy+n+,cmp);
        energy[i]=heavy[i]+heavy[i-];
    }
    ;i<=n-;i++)
     total+=energy[i];
     printf("%d",total);
     ;
}

那么在这里就用到了堆(不懂堆的小朋友:详见->《堆》)

那么思路是这样的:

首先这个问题比较模糊,所以我们可以将它的问题描述转化一下:给了n个节点(即果子堆数),每一个节点有一个权值w[i](每一个堆的果子个数),我们要做的就是将其中的两个合并为一棵树, 假设每一个节点到根节点的距离为d[i],那么我们就要使得最终的∑(w[i]*d[i])最小(不懂∑的请点这里->《希腊字母在数学计算中表示的含义》)。那么具体的方法就是这样的:

  1. 从这个森林里面取出两个权值最小的节点。
  2. 将他们的权值相加,得到一个新的子树,然后删除原子树。
  3. 将当前得到的新子树插入到森林中然后维护。
  4. 不断重复1~3直到整个森林里面只有一棵树为止。

所以这里最重要的其实就是两个操作:取出get()和插入put()。

但是由于取出只能取出根节点,二我们要取出的还必须是权值最小的节点,所以我们要制作一个小根堆。

下面附上代码:

#include<iostream>
#include<cstdio>
using namespace std;
int heap_size,n;
];
void swip(int &a,int &b)
{
    ;a=b;b=t;
}
void put(int d)
{
    int now,next;
    heap[++heap_size]=d;
    now=heap_size;
    )
    {
        next=now>>;
        if(heap[now]>=heap[next]) return;
        swap(heap[now],heap[next]);
        now=next;
    }
}
int get()
{
    int now,next,res;
    res=heap[];
    heap[]=heap[heap_size--];
    now=;
    <=heap_size)
    {
        next=now*;
        ]<heap[next]) next++;
        if(heap[now]<heap[next]) return res;
        swap(heap[now],heap[next]);
        now=next;
    }
    return res;
}
void work()
{
    ;
    scanf("%d",&n);
    ;i<=n;i++)
    {
        scanf("%d",&x);
        put(x);
    }
    ;i<n;i++)
    {
        x=get();
        y=get();
        ans+=x+y;
        put(x+y);
    }
    printf("%d",ans);
}
int main()
{
    work();
    ;
}

[luoguP1090][Noip2004]合并果子的更多相关文章

  1. NC16663 [NOIP2004]合并果子

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

  2. NOIP2004合并果子

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

  3. 合并果子(NOIP2004)

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

  4. [Noip2004][Day ?][T?]合并果子(?.cpp)

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

  5. [NOIP2004] 提高组 洛谷P1090 合并果子

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

  6. 加强版:合并果子[NOIP2004]

    题目 链接:https://ac.nowcoder.com/acm/contest/26887/1001 来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 131072K, ...

  7. 合并果子 (codevs 1063) 题解

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

  8. 代码源 每日一题 分割 洛谷 P6033合并果子

    ​ 题目链接:切割 - 题目 - Daimayuan Online Judge 数据加强版链接: [NOIP2004 提高组] 合并果子 加强版 - 洛谷 题目描述 有一个长度为 ∑ai 的木板,需要 ...

  9. 【noip 2004】 合并果子

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

随机推荐

  1. 很有用的PHP笔试题系列二

    1.如何用php的环境变量得到一个网页地址的内容?ip地址又要怎样得到? Gethostbyname() echo $_SERVER ["PHP_SELF"];echo $_SER ...

  2. List< >泛型集合

    //一旦你确定了泛型的类型,集合里的元素已经有了确切的定义 List<int> list = new List<int>();  //创建泛型集合的对象 list.Add(); ...

  3. Windows无法启动MySQL服务,错误1067

    问题:mysql服务启动异常 找出原因 检查D:\mysql-5.6.20-winx64\data目录下client-02.err文件的错误信息(以err为后缀名的文件是mysql的日志文件) 修改配 ...

  4. 04.Continue,和三元表达式的学习

    立即结束本次循环,判断循环条件,如果成立,则进入下一次循环,否则退出循环. 举例:运动员跑步喝水的例子 比如:我编写代码的时候,上个厕所,回来继续写代码 练习1: namespace _09.练习02 ...

  5. scss-颜色运算符

    scss允许使用颜色分量以及算术运算和任何颜色表达式返回颜色值. 例如scss代码: $color1: #333399; $color2: #CC3399; p{ color: $color1 + $ ...

  6. 关于asp.net网站中web.config的配置

    1.Session生命周期配置 2.连接默认错误页 实例说明:在一些网站中,当网络地址发生错误时,通常会自动跳转到一个页面,并在该页面显示错误信息,此功能也可以通过配置web.config文件配置实现 ...

  7. 有关ie9 以下不支持placeholder属性以及获得焦点placeholder的移除

    (一)placeholder 属性起到提示客户输入信息作用 (二)ie9以下出问题了 placeholder不支持 (三)解决办法 先贴html 加上jquery代码就可以了 关于表单获取焦点,chr ...

  8. Linux虚拟系统安装——Ubuntu18.04 & CentOS6.5

    Linux虚拟系统安装--Ubuntu18.04 & CentOS6.5 摘要:Linux简介.虚拟系统安装.系统备份与文件介绍 1. Linux简介 (1)1968年,MIT.Bell实验室 ...

  9. 通知:即日起本博客暂停更新,请移步至yanxin8.com获取最新文章

    通知:即日起本博客暂停更新,请移步至yanxin8.com与博主交流及获取最新文章

  10. IS Decisions如何帮助企业提高安全标准

    PCI DSS标准有什么要求? 简单地说,PCI DSS要求最高级别的网络安全性.这一标准如今广泛应用于需要存储.管理.传输客户(或持卡人)个人数据的行业和领域. 施行严格的访问监控措施 为了保证关键 ...