洛谷 P1131 [ZJOI2007]时态同步

小Q在电子工艺实习课上学习焊接电路板。一块电路板由若干个元件组成,我们不妨称之为节点,并将其用数字1,2,3….进行标号。电路板的各个节点由若干不相交的导线相连接,且对于电路板的任何两个节点,都存在且仅存在一条通路(通路指连接两个元件的导线序列)。

在电路板上存在一个特殊的元件称为“激发器”。当激发器工作后,产生一个激励电流,通过导线传向每一个它所连接的节点。而中间节点接收到激励电流后,得到信息,并将该激励电流传向与它连接并且尚未接收到激励电流的节点。最终,激烈电流将到达一些“终止节点”――接收激励电流之后不再转发的节点。

激励电流在导线上的传播是需要花费时间的,对于每条边e,激励电流通过它需要的时间为\(t_e\)
,而节点接收到激励电流后的转发可以认为是在瞬间完成的。现在这块电路板要求每一个“终止节点”同时得到激励电路――即保持时态同步。由于当前的构造并不符合时态同步的要求,故需要通过改变连接线的构造。目前小Q有一个道具,使用一次该道具,可以使得激励电流通过某条连接导线的时间增加一个单位。请问小Q最少使用多少次道具才可使得所有的“终止节点”时态同步?

就是给你一个有根树,让你每次给一个边加1直到根到所有叶子节点的距离相同,问最少操作

这题好像很水的样子QAQ

考虑树上dp,设\(f_u\)表示u的子树时态同步的最小代价,\(dis_u\)表示u的子树时态同步后的子树边权和

然后写出状态转移方程

\[\begin{cases} dis_u=max_{v\in son(u)}dis_v+dist(u,v)\\ f_u=\sum_{v\in son(u)}f_v+dis_u-dis_v-dist(u,v) \end{cases}\]

\(dist(u,v)\)表示u到v的距离

因为边权只能加1,所以我们只能找最大的边来作为相同的边,也就是儿子的dis加上这条边取最大值

然后贡献的话首先要把所有儿子变相同的加起来,然后再把比dis小的边的贡献算上就可以了

Code

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <vector>
const int N = 5e5;
using namespace std;
struct node
{
    int to,cost;
};
int n,s;
long long f[N + 5],dis[N + 5];
vector <node> d[N + 5];
void dfs(int u,int fa)
{
    vector <node>::iterator it;
    int maxx = 0;
    for (it = d[u].begin();it != d[u].end();it++)
    {
        int v = (*it).to,w = (*it).cost;
        if (v == fa)
            continue;
        maxx = max(maxx,w);
        dfs(v,u);
        f[u] += f[v];
        dis[u] = max(dis[v] + w,dis[u]);
    }
    for (it = d[u].begin();it != d[u].end();it++)
    {
        int v = (*it).to,w = (*it).cost;
        if (v == fa)
            continue;
        f[u] += dis[u] - dis[v] - w;
    }
}
int main()
{
    scanf("%d%d",&n,&s);
    int u,v,w;
    for (int i = 1;i < n;i++)
    {
        scanf("%d%d%d",&u,&v,&w);
        d[u].push_back((node){v,w});
        d[v].push_back((node){u,w});
    }
    dfs(s,0);
    cout<<f[s]<<endl;
    return 0;
}
posted @ 2020-06-08 21:11  eee_hoho  阅读(97)  评论(0编辑  收藏  举报