标签:mat 说明 输入 ret 遍历 最小 alt 医院 splay
设有一棵二叉树,如图:
其中,圈中的数字表示结点中居民的人口。圈边上数字表示结点编号,现在要求在某个结点上建立一个医院,使所有居民所走的路程之和为最小,同时约定,相邻接点之间的距离为 11。如上图中,若医院建在1 处,则距离和 =4+12+2\times20+2\times40=136=4+12+2×20+2×40=136;若医院建在 33 处,则距离和 =4\times2+13+20+40=81=4×2+13+20+40=81。
第一行一个整数 nn,表示树的结点数。
接下来的 nn 行每行描述了一个结点的状况,包含三个整数 w, u, vw,u,v,其中 ww 为居民人口数,uu 为左链接(为 00 表示无链接),vv 为右链接(为 00 表示无链接)。
一个整数,表示最小距离和。
输入 #1复制
5
13 2 3
4 0 0
12 4 5
20 0 0
40 0 0
输出 #1复制
81
这题是一题典型的Floyd算法,我们使用一个全图进行记录最短路径,最后进行遍历,当以i为基站时,各个顶点到它的最短路*权值。
#include <iostream>
#include <algorithm>
#include <cstdio>
#define M 0x7fffffff
using namespace std;
int W[110] = {0}, path[110][110], n, a, b;
int main() {
fill(path[0], path[0] + 110 * 110, 99999);
scanf("%d", &n);
// 存储图
for(int i = 1; i <= n; i++){
scanf("%d%d%d", &W[i], &a, &b);
path[i][i] = 0;
if(a > 0) path[i][a] = path[a][i] = 1;
if(b > 0) path[i][b] = path[b][i] = 1;
}
// 使用Floyd算法求每两点的最短路
for(int k = 1; k <= n; k++)
for(int i = 1; i <= n; i++)
for(int j = 1; j <= n; j++)
if(path[i][k] + path[k][j] < path[i][j])
path[i][j] = path[i][k] + path[k][j];
// 求最小顶点
int minTotal = M;
for(int i = 1; i <= n; i++) {
int total = 0;
for(int j = 1; j <= n; j++)
total += W[j] * path[i][j];
if(total < minTotal) minTotal = total;
}
printf("%d", minTotal);
return 0;
}
标签:mat 说明 输入 ret 遍历 最小 alt 医院 splay
原文地址:https://www.cnblogs.com/littlepage/p/13074022.html