蓝桥杯模拟赛——村庄通电

题目介绍

  • 问题描述
    2015年,全中国实现了户户通电。作为一名电力建设者,小明正在帮助一带一路上的国家
    通电。
    这一次,小明要帮助 n 个村庄通电,其中 1 号村庄正好可以建立一个发电站,所发的电足
    够所有村庄使用。
    现在,这 n 个村庄之间都没有电线相连,小明主要要做的是架设电线连接这些村庄,
    使得所有村庄都直接或间接的与发电站相通。
    小明测量了所有村庄的位置(坐标)和高度,如果要连接两个村庄,小明需要花费两个
    村庄之间的坐标距离加上高度差的平方,形式化描述为坐标为 (x_1, y_1) 高度为 h_1 的
    村庄与坐标为 (x_2, y_2) 高度为 h_2 的村庄之间连接的费用为
    sqrt((x_1-x_2)(x_1-x_2)+(y_1-y_2)(y_1-y_2))+(h_1-h_2)*(h_1-h_2)。
    在上式中 sqrt 表示取括号内的平方根。请注意括号的位置,高度的计算方式与横纵坐标的
    计算方式不同。
    由于经费有限,请帮助小明计算他至少要花费多少费用才能使这 n 个村庄都通电。
    输入格式
      输入的第一行包含一个整数 n ,表示村庄的数量。
      接下来 n 行,每个三个整数 x, y, h,分别表示一个村庄的横、纵坐标和高度,
    其中第一个村庄可以建立发电站。
    输出格式
      输出一行,包含一个实数,四舍五入保留 2 位小数,表示答案。
    样例输入
    4
    1 1 3
    9 9 7
    8 8 6
    4 5 4
    样例输出
    17.41
    评测用例规模与约定
      对于 30% 的评测用例,1 <= n <= 10;
      对于 60% 的评测用例,1 <= n <= 100;
      对于所有评测用例,1 <= n <= 1000,0 <= x, y, h <= 10000。

import java.util.Scanner;

public class Main {
	static class Node {
		int x;
		int y;
		int h;
	}

	public static void main(String[] args) {
		//输入
		Node[] nodes = new Node[1002];
		 
		Scanner sc = new Scanner(System.in);
		int n = sc.nextInt();
		for (int i = 1; i <= n; i++) {
			nodes[i]=new Node();
			nodes[i].x = sc.nextInt();
			nodes[i].y = sc.nextInt();
			nodes[i].h = sc.nextInt();
		}
		sc.close();
		//初始化数组
		double[][] map = new double[n + 2][n + 2];
		double MAX = 0x7f7f7f7f;
		//初始化数组
		for (int i = 1; i <= n; i++) {
			for (int j = 1; j <=n; j++) {
				 map[i][j]=MAX;			
			}
		}
		//形成邻接矩阵
		for (int i = 1; i <= n-1; i++) {
			for (int j = i + 1; j <= n; j++) {
			double	x = (nodes[i].x - nodes[j].x) * (nodes[i].x - nodes[j].x);
			double	y = (nodes[i].y - nodes[j].y) * (nodes[i].y - nodes[j].y);
			double	h = (nodes[i].h - nodes[j].h) * (nodes[i].h - nodes[j].h);
				double temp=Math.sqrt(x+y)+h;
				map[i][j]=Math.min(map[i][j],temp );
				map[j][i]=map[i][j];
			}
		}
		//使用Prim算法求最小生成树
		int[] visited = new int[n+1];//标记已经访问过的村庄
		visited[1] = 1 ;//从第一个村庄开始
		int h1 = -1;//标记通电的上流村庄
		int h2 = -1;//标记通电的下流村庄
		double min = MAX;
		double cnt = 0.0;//记录总花费
		for(int i = 1;i<n;i++)//需要找n-1条边(n个村庄,n-1条边)
		{
			//每执行一次这个for循环就会找到一条新的最小花费的边
			for(int p = 1;p<=n;p++)
			{ 
				for(int q = 1; q<=n;q++)
				{
					//从已经通电的村庄向没通电的村庄探索,并且找到通往下一个花费最少的村庄
					if(visited[p] ==1&&visited[q]==0&&map[p][q]<min)
					{
						min = map[p][q];
						h1 = p;
						h2 = q;
					}
				}
			}
			visited[h2] = 1;//标记新的村庄已经访问过
			cnt+=min;//累计花费的费用
			min = MAX;//每次找到一个村庄,都要重新把min赋一下值
		}
		System.out.printf("%.2f",cnt);
	}

}

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值