【bzoj3573】【Hnoi2014】米特运输(树形dp)

在D星球上,一种名为米特的神秘物质作为主要能源,需通过特定算法确保各城市间有效且安全地调度存储。面对技术约束,本文介绍了一个优化方案,旨在通过调整部分城市的米特储存器容量来实现最高效的能源分配。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Description

米特是D星球上一种非常神秘的物质,蕴含着巨大的能量。在以米特为主要能源的D星上,这种米特能源的运输和储存一直是一个大问题。D星上有N个城市,我们将其顺序编号为1到N,1号城市为首都。这N个城市由N-1条单向高速通道连接起来,构成一棵以1号城市(首部)为根的树,高速通道的方向由树中的儿子指向父亲。树按深度分层:根结点深度为0,属于第1层;根结点的子节点深度为1,属于第2层;依此类推,深度为i的结点属于第i+l层。建好高速通道之后,D星人开始考虑如何具体地储存和传输米特资源。由于发展程度不同,每个城市储存米特的能力不尽相同,其中第i个城市建有一个容量为A[i]的米特储存器。这个米特储存器除了具有储存的功能,还具有自动收集米特的能力。如果到了晚上六点,有某个储存器处于未满的状态,它就会自动收集大气中蕴含的米特能源,在早上六点之前就能收集满;但是,只有在储存器完全空的状态下启动自动收集程序才是安全的,未满而又非空时启动可能有安全隐患。早上六点到七点间,根节点城市(1号城市)会将其储存器里的米特消耗殆尽。根节点不会自动搜集米特,它只接受子节点传输来的米特。早上七点,城市之间启动米特传输过程,传输过程逐层递进:先是第2层节点城市向第1层(根节点城市,即1号城市)传输,直到第1层的储存器满或第2层的储存器全为空;然后是第3层向第2层传输,直到对于第2层的每个节点,其储存器满或其予节点(位于第3层)的储存器全为空;依此类推,直到最后一层传输完成。传输过程一定会在晚上六点前完成。
由于技术原因,运输方案需要满足以下条件:
(1)不能让某个储存器到了晚上六点传输结束时还处于非空但又未满的状态,这个时候储存器仍然会启动自动收集米特的程序,而给已经储存有米特的储存器启动收集程序可能导致危险,也就是说要让储存器到了晚上六点时要么空要么满;
(2)关于首都——即1号城市的特殊情况, 每天早上六点到七点间1号城市中的米特储存器里的米特会自动被消耗殆尽,即运输方案不需要考虑首都的米特怎么运走;
(3)除了1号城市,每个节点必须在其子节点城市向它运输米特之前将这座城市的米特储存器中原本存有的米特全部运出去给父节点,不允许储存器中残存的米特与外来的米特发生混合;
(4)运向某一个城市的若干个来源的米特数量必须完全相同,不然,这些来源不同的米特按不同比例混合之后可能发生危险。
现在D星人已经建立好高速通道,每个城市也有了一定储存容量的米特储存器。为了满足上面的限制条件,可能需要重建一些城市中的米特储存器。你可以,也只能,将某一座城市(包括首都)中屎来存在的米特储存器摧毁,再新建一座任意容量的新的米特储存器,其容量可以是小数(在输入数据中,储存器原始容量是正整数,但重建后可以是小数),不能是负数或零,使得需要被重建的米特储存器的数目尽量少。

Input

第一行是一个正整数N,表示城市的数目。接下来N行,每行一个正整数,其中的第i行表示第i个城市原来存在的米特储存器的容量。再接下来是N-I行,每行两个正整数a,b表示城市b到城市a有一条高速通道(a≠b)。
N<500000,A[j]<10^8

Output

输出文件仅包含一行,一个整数,表示最少的被重建(即修改储存器容量)的米特储存器的数目。

Sample Input

5

5

4

3

2

1

1 2

1 3

2 4

2 5

Sample Output

3

HINT

一个最优解是将A[1]改成8,A[3]改成4,A[5]改成2。这样,2和3运给1的量相等,4和5运给2的量相等,且每天晚上六点的时候,1,2满,3,4,5空,满足所有限制条件。

题意:
给定一棵树和每个点的权值,问最少改动多少个点的权值使得:
1.每个点的所有儿子权值相等
2.每个点的权值等于所有儿子权值和

题解:
本题的关键点:如果一个节点的值确定了,所有结点的值都确定了。
所以枚举每个节点不变,求出根节点的值,选取值相同最多的那个方案,就可以使改动最小,次数为n-该值出现次数。
有个问题:根节点的值太大了,有可能爆long long,用map或对数。取对数精度在1e-5以下都可以。
代码如下:

#include<iostream>
#include<stdio.h>
#include<algorithm>
#include<string.h>
#include<math.h>
#define ll long long
#define inf 0x7f7f7f7f
#define N 500005
using namespace std;
struct nod
{
    int v,nex;
    nod(){}
    nod(int a,int b):v(a),nex(b){}
}e[N<<1];
int cnt,hd[N];
void add(int u,int v)
{
    e[++cnt]=nod(v,hd[u]);hd[u]=cnt;
    e[++cnt]=nod(u,hd[v]);hd[v]=cnt;
}
void read(int &x)
{
    x=0;
    char c=getchar();
    while(c<'0' || c>'9') c=getchar();
    while(c<='9' && c>='0')
    {
        x=x*10+c-'0';
        c=getchar();
    }
}
int n,x,y,d[N],a[N];
double s[N],val[N];
void dfs(int u,int fa)
{
    for(int i=hd[u];i;i=e[i].nex)
    {
        int nex=e[i].v;
        if(nex==fa) continue;
        s[nex]=s[u]+log(d[u]); 
        dfs(nex,u);
    } 
}
int main()
{
    read(n);
    for(int i=1;i<=n;i++) read(a[i]);
    for(int i=1;i<n;i++)
    {
        read(x),read(y);
        add(x,y);
        d[x]++,d[y]++;
    }
    for(int i=2;i<=n;i++) d[i]--;
    s[1]=0.0;
    dfs(1,0);
    for(int i=1;i<=n;i++) val[i]=s[i]+log(a[i]);
    sort(val+1,val+n+1);
    int tot=1,ans=0;
    for(int i=2;i<=n;i++)
    {
        if(val[i]-val[i-1]<1e-6) tot++;
        else
        {
            ans=max(ans,tot);
            tot=1;
        }
    }
    ans=max(ans,tot);
    printf("%d\n",n-ans);
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值