【模板】树的重心 洛谷P1364 医院设置

P1364 医院设置

题目描述

设有一棵二叉树,如图:

                                         

其中,圈中的数字表示结点中居民的人口。圈边上数字表示结点编号,现在要求在某个结点上建立一个医院,使所有居民所走的路程之和为最小,同时约定,相邻接点之间的距离为l。如上图中,

若医院建在1 处,则距离和=4+12+2*20+2*40=136;若医院建在3 处,则距离和=4*2+13+20+40=81……

输入输出格式

输入格式:

第一行一个整数n,表示树的结点数。(n≤100)

接下来的n行每行描述了一个结点的状况,包含三个整数,整数之间用空格(一个或多个)分隔,其中:第一个数为居民人口数;第二个数为左链接,为0表示无链接;第三个数为右链接。

输出格式:

一个整数,表示最小距离和。

输入输出样例

输入样例#1:

5
13 2 3
4 0 0
12 4 5
20 0 0
40 0 0

输出样例#1:

81

【题解】
随便找个跟DFS一下,第一个子树权值 * 2>=总权值的点即为重心(为什么?我也不知道)
然后再DFS一下就好了

 1 #include <bits/stdc++.h>
 2 inline void read(int &x)
 3 {
 4     x = 0;char ch = getchar();char c = ch;
 5     while(ch > ‘9‘ || ch < ‘0‘)c = ch, ch = getchar();
 6     while(ch <= ‘9‘ && ch >= ‘0‘)x = x * 10 + ch - ‘0‘, ch = getchar();
 7     if(c == ‘-‘)x = -x;
 8 }
 9
10 const int INF = 0x3f3f3f3f;
11 const int MAXN = 500 + 10;
12
13 struct Edge
14 {
15     int u,v,next;
16 }edge[MAXN + 1];
17 int head[MAXN],cnt,root,b[MAXN],n,sum,w[MAXN],ans;
18
19 void insert(int a, int b){edge[++cnt] = Edge{a,b,head[a]};head[a] = cnt;}
20
21 int dfs1(int u)
22 {
23     register int tmp = w[u];
24     for(register int pos = head[u];pos;pos = edge[pos].next)
25     {
26         int v = edge[pos].v;
27         if(!b[v])
28         {
29             b[v] = true;
30             tmp += dfs1(v);
31             if(tmp * 2 >= sum && !root)
32                 root = u;
33             if(root)
34                 return 0;
35         }
36     }
37     return tmp;
38 }
39
40 void dfs2(int u, int step)
41 {
42     ans += w[u] * step;
43     for(register int pos = head[u];pos;pos = edge[pos].next)
44     {
45         int v = edge[pos].v;
46         if(!b[v])
47         {
48             b[v] = 1;
49             dfs2(v, step + 1);
50         }
51     }
52 }
53
54 int main()
55 {
56     read(n);
57     register int tmp1,tmp2,tmp3;
58     for(register int i = 1;i <= n;++ i)
59     {
60         read(tmp1);read(tmp2);read(tmp3);
61         if(tmp2)
62             insert(i,tmp2),insert(tmp2,i);
63         if(tmp3)
64             insert(i,tmp3),insert(tmp3,i);
65         w[i] = tmp1;
66         sum += tmp1;
67     }
68     b[1] = 1;
69     dfs1(1);
70     memset(b, 0, sizeof(b));
71     b[root] = 1;
72     dfs2(root, 0);
73     printf("%d", ans);
74     return 0;
75 }

时间: 2024-10-08 20:19:59

【模板】树的重心 洛谷P1364 医院设置的相关文章

洛谷P1364 医院设置

P1364 医院设置 题目描述 设有一棵二叉树,如图: 其中,圈中的数字表示结点中居民的人口.圈边上数字表示结点编号,现在要求在某个结点上建立一个医院,使所有居民所走的路程之和为最小,同时约定,相邻接点之间的距离为l.如上图中, 若医院建在1 处,则距离和=4+12+2*20+2*40=136:若医院建在3 处,则距离和=4*2+13+20+40=81…… 输入输出格式 输入格式: 第一行一个整数n,表示树的结点数.(n≤100) 接下来的n行每行描述了一个结点的状况,包含三个整数,整数之间用空

洛谷 P1364 医院设置

题目描述 设有一棵二叉树,如图: 其中,圈中的数字表示结点中居民的人口.圈边上数字表示结点编号,现在要求在某个结点上建立一个医院,使所有居民所走的路程之和为最小,同时约定,相邻接点之间的距离为l.如上图中, 若医院建在1 处,则距离和=4+12+2*20+2*40=136:若医院建在3 处,则距离和=4*2+13+20+40=81…… 输入输出格式 输入格式: 第一行一个整数n,表示树的结点数.(n≤100) 接下来的n行每行描述了一个结点的状况,包含三个整数,整数之间用空格(一个或多个)分隔,

洛谷P1364 医院设置(Floyd)

题目描述 设有一棵二叉树,如图: 其中,圈中的数字表示结点中居民的人口.圈边上数字表示结点编号,现在要求在某个结点上建立一个医院,使所有居民所走的路程之和为最小,同时约定,相邻接点之间的距离为l.如上图中, 若医院建在1 处,则距离和=4+12+2*20+2*40=136:若医院建在3 处,则距离和=4*2+13+20+40=81…… 输入输出格式 输入格式: 第一行一个整数n,表示树的结点数.(n≤100) 接下来的n行每行描述了一个结点的状况,包含三个整数,整数之间用空格(一个或多个)分隔,

luogu P1364 医院设置

P1364 医院设置 题目描述 设有一棵二叉树,如图: 其中,圈中的数字表示结点中居民的人口.圈边上数字表示结点编号,现在要求在某个结点上建立一个医院,使所有居民所走的路程之和为最小,同时约定,相邻接点之间的距离为l.如上图中, 若医院建在1 处,则距离和=4+12+2*20+2*40=136:若医院建在3 处,则距离和=4*2+13+20+40=81-- 输入输出格式 输入格式: 第一行一个整数n,表示树的结点数.(n≤100) 接下来的n行每行描述了一个结点的状况,包含三个整数,整数之间用空

P1364 医院设置

题目描述 设有一棵二叉树,如图: 其中,圈中的数字表示结点中居民的人口.圈边上数字表示结点编号,现在要求在某个结点上建立一个医院,使所有居民所走的路程之和为最小,同时约定,相邻接点之间的距离为l.如上图中, 若医院建在1 处,则距离和=4+12+2*20+2*40=136:若医院建在3 处,则距离和=4*2+13+20+40=81-- 输入输出格式 输入格式: 第一行一个整数n,表示树的结点数.(n≤100) 接下来的n行每行描述了一个结点的状况,包含三个整数,整数之间用空格(一个或多个)分隔,

树状数组模板1(例题洛谷P3374)(单点修改+区间查询)

例题:https://www.luogu.org/problem/show?pid=3374 程序: #include<iostream> #include<cstdio> #include<algorithm> using namespace std; int n,m,x,y,ch,f[600000],a[600000]; int lowbit(int x) { return (x&-x); } //单点修改 void update(int x,int k)

【模板】ST表 洛谷P1816 忠诚

P1816 忠诚 题目描述 老管家是一个聪明能干的人.他为财主工作了整整10年,财主为了让自已账目更加清楚.要求管家每天记k次账,由于 管家聪明能干,因而管家总是让财主十分满意.但是由于一些人的挑拨,财主还是对管家产生了怀疑.于是他决定用一种特别的方法来判断管家的忠诚,他把每次的 账目按1,2,3…编号,然后不定时的问管家问题,问题是这样的:在a到b号账中最少的一笔是多少?为了让管家没时间作假他总是一次问多个问题. 输入输出格式 输入格式: 输入中第一行有两个数m,n表示有m(m<=100000

AC日记——【模板】二分图匹配 洛谷 P3386

题目背景 二分图 题目描述 给定一个二分图,结点个数分别为n,m,边数为e,求二分图最大匹配数 输入输出格式 输入格式: 第一行,n,m,e 第二至e+1行,每行两个正整数u,v,表示u,v有一条连边 输出格式: 共一行,二分图最大匹配 输入输出样例 输入样例#1: 1 1 1 1 1 输出样例#1: 1 说明 n,m<=1000,1<=u<=n,1<=v<=m 因为数据有坑,可能会遇到v>m的情况.请把v>m的数据自觉过滤掉. 算法:二分图匹配 思路: 二分图模

P1364 医院设置(树型结构)

传送门闷闷闷闷闷闷 ~~放一个可爱的输入框.~~ 考虑在O(n)的时间内求数以每个节点为医院的距离和. \(设想一下,如果我们已知以1为根节点的距离和f[1],如何求出子节点呢?\) 当医院从1转换到1的儿子节点2 一.那么2为根的子树节点到医院的距离都减少1 二.其余节点到医院的距离都增加1 所以得出方程\(f[v]=f[u]-size[v]+(size[1]-size[v])\) 其中\(size[i]\)表示以i为根的子树节点数的权值和 #include <bits/stdc++.h>