每周题解:牛的旅行

题目描述

牛的旅行

农民John的农场里有很多牧区。有的路径连接一些特定的牧区。一片所有连通的牧区称为一个牧场。但是就目前而言,你能看到至少有两个牧区不连通。

现在,John想在农场里添加一条路径 ( 注意,恰好一条 )。对这条路径有这样的限制:一个牧场的直径就是牧场中最远的两个牧区的距离 ( 本题中所提到的所有距离指的都是最短的距离 )。

考虑如下的两个牧场,图1是有 5 5 5个牧区的牧场,牧区用*表示,路径用直线表示。每一个牧区都有自己的坐标:
在这里插入图片描述

图1所示的牧场的直径大约是 12.07106 12.07106 12.07106, 最远的两个牧区是 A A A E E E,它们之间的最短路径是 A → B → E A\to B\to E ABE。 这两个牧场都在John的农场上。John将会在两个牧场中各选一个牧区,然后用一条路径连起来,使得连通后这个新的更大的牧场有最小的直径。

注意,如果两条路径中途相交,我们不认为它们是连通的。只有两条路径在同一个牧区相交,我们才认为它们是连通的。

现在请你编程找出一条连接两个不同牧场的路径,使得连上这条路径后,所有牧场(生成的新牧场和原有牧场)中直径最大的牧场的直径尽可能小

输入格式

第 1 行:一个整数 N N N ( 1 ≤ N ≤ 150 1 ≤ N ≤ 150 1N150), 表示牧区数;
第 2 到 N + 1 N+1 N+1 行:每行两个整数 X X X Y Y Y ( 0 ≤ X , Y ≤ 100000 0≤ X,Y≤ 100000 0XY100000 ), 表示 N N N个牧区的坐标,每个牧区的坐标都是不一样的。
N + 2 N+2 N+2 行到第 2 × N + 1 2\times N+1 2×N+1 行:每行包括 N N N个数字 ( 0 0 0 1 1 1 ) 表示一个对称邻接矩阵。 例如,题目描述中的两个牧场的矩阵描述如下:

  A B C D E F G H 
A 0 1 0 0 0 0 0 0 
B 1 0 1 1 1 0 0 0 
C 0 1 0 0 1 0 0 0 
D 0 1 0 0 1 0 0 0 
E 0 1 1 1 0 0 0 0 
F 0 0 0 0 0 0 1 0 
G 0 0 0 0 0 1 0 1 
H 0 0 0 0 0 0 1 0

输出格式

只有一行,包括一个实数,表示所求答案。数字保留六位小数。

样例 #1

样例输入 #1

8
10 10
15 10
20 10
15 15
20 15
30 15
25 10
30 10
01000000
10111000
01001000
01001000
01110000
00000010
00000101
00000010

样例输出 #1

22.071068

算法思想

根据题目描述,测试样例可以得到这样一张图:
在这里插入图片描述
其中 A , B , C , D , E A,B,C,D,E A,B,C,D,E属于一个牧场, F , G , H F,G,H F,G,H属于另外一个牧场。现在要将两个牧场用一条路径连起来,使得连通后这个新的更大的牧场有最小的直径。

用一条路径连接后,要求所有牧场(生成的新牧场和原有牧场)中直径最大的牧场,使其直径尽可能小。那么直径最大的牧场无非有两种情况:

  • 最大牧场是原有牧场
  • 最大牧场是生成的新牧场

可以确定结果一定是大于等于所有原有牧场的直径的,并且生成的新牧场的直径必须尽可能的小,所以答案应该取这两者的最大值。

那么算法的基本思想:

  • 首先求出原有牧场中,任意两个的牧区之间的最短距离,可以用「Floyd」算法完成
  • 然后求出在同一牧场中,到牧区 i i i的最远距离 m a x d [ i ] maxd[i] maxd[i],同时打擂台求其中的最大值 r e s 1 res1 res1,就是原有牧场的直径最大值。
  • 其次,枚举一条连接两个不同牧场的路径,求出连接后的牧场直径,即 m a x d [ i ] + x + m a x d [ j ] maxd[i] + x + maxd[j] maxd[i]+x+maxd[j],其中 i i i j j j属于不同牧场, x x x表示牧区 i i i j j j的距离,那么其中最小的就是生成的新牧场的直径 r e s 2 res2 res2
  • 最后, r e s 1 res1 res1 r e s 2 res2 res2的最大值就是答案。

时间复杂度

本题的核心是使用Floyd算法计算任意两个牧区之间的最短路,时间复杂度为 O ( n 3 ) O(n^3) O(n3)

代码实现

#include <bits/stdc++.h>
using namespace std;
typedef pair<double,double> PDD;
const int N = 155;
const double INF = 1e20;
int n;
PDD p[N];
char g[N][N];
double d[N][N], maxd[N]; //maxd[i]表示在一个牧场中距离牧区i的最远距离
double get_dis(PDD a, PDD b)
{double dx = a.first - b.first;double dy = a.second - b.second;return sqrt(dx * dx + dy * dy);
}
int main()
{cin >> n;for(int i = 0; i < n; i ++) cin >> p[i].first >> p[i].second;for(int i = 0; i < n; i ++) cin >> g[i];//初始化状态数组for(int i = 0; i < n; i ++)for(int j = 0; j < n; j ++)if(i == j) d[i][j] = 0; //同一个点else if(g[i][j] == '1') d[i][j] = get_dis(p[i], p[j]); //相邻计算距离else d[i][j] = INF;//Floyd计算任意两点之间的距离for(int k = 0; k < n; k ++)for(int i = 0; i < n; i ++)for(int j = 0; j < n; j ++)d[i][j] = min(d[i][j], d[i][k] + d[k][j]);double res1 = 0;//计算每个牧区连通的最远牧区的距离for(int i = 0; i < n; i ++){for(int j = 0; j < n; j ++){if(d[i][j] < INF / 2) //i和j是连通的maxd[i] = max(maxd[i], d[i][j]);}res1 = max(res1, maxd[i]); //求所有牧场的最大直径}//枚举一条连接两个不同牧场的路径double res2 = INF; //求连通两个牧场后,直径的最小值for(int i = 0; i < n; i ++)for(int j = 0 ; j < n; j ++){if(d[i][j] > INF / 2) //两个牧区不连通{double x = get_dis(p[i], p[j]);res2 = min(res2, maxd[i] + x + maxd[j]);}}printf("%.6lf", max(res1, res2));return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/706505.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

docker镜像容器常用命令

常用基础命令1、docker info #查看docker版本等信息 2、docker search jenkins #搜索jenkins镜像 3、docker history nginx #查看镜像中各层内容及大小,每层对应的dockerfile中的一条指令。 4、docker network ls #显示当前主机上的所有网络 5、docker logs nginx …

2025秋招Java还是c++?

一、我的编程经 说说我的编程经历&#xff0c;在C和Java之间我经历了几个阶段&#xff1a; 大学期间&#xff0c;我浅尝辄止地学习了一段时间的Java&#xff0c;但后来放弃了&#xff0c;开始学习C/C。本科毕业后&#xff0c;我选择攻读硕士学位&#xff0c;并一直专注于C的学…

C# 快速排序(QuickSort)

QuickSort是一种基于分而治之算法的排序算法&#xff0c;它选择一个元素作为主元&#xff0c;并通过将主元放置在已排序数组中的正确位置&#xff0c;围绕所选主元对给定数组进行分区。 快速排序是如何工作的&#xff1f; QuickSort中的关键过程是partition()。分区的…

[数据集][目标检测]电力场景电力目标检测数据集VOC+YOLO格式476张5类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;476 标注数量(xml文件个数)&#xff1a;476 标注数量(txt文件个数)&#xff1a;476 标注类别…

springboot企业员工考勤系统-计算机毕业设计源码57302

摘要 由于数据库和数据仓库技术的快速发展&#xff0c;企业员工考勤系统建设越来越向模块化、智能化、自我服务和管理科学化的方向发展。员工管理系统对处理对象和服务对象&#xff0c;自身的系统结构&#xff0c;处理能力&#xff0c;都将适应技术发展的要求发生重大的变化。 …

Postman基础功能-接口返回值获取

大家好&#xff0c;之前给大家分享关于Postman的接口关联&#xff0c;我们平时在做接口测试时&#xff0c;请求接口返回的数据都是很复杂的 JSON 数据&#xff0c;有着多层嵌套&#xff0c;这样的数据层级在 Postman 中要怎么获取呢&#xff1f; 接下来给大家展示几个获取 JSO…

2024 年适用于 Mac 的 5 大最佳免费数据恢复工具

一个常见的误解是&#xff0c;数据恢复总是很昂贵。实际上&#xff0c;您可以在 2024 年下载许多适用于 Mac 的免费数据恢复软件工具&#xff0c;并使用它们来恢复丢失的数据&#xff0c;而无需将 Mac 交给数据恢复专业人员&#xff0c;他们保证会向您收取一小笔费用他们的服务…

云原生 初识Kubernetes的理论基础

一、k8s 的由来及其技术运用 1.1 k8s的简介 Kubernetes&#xff0c;词根源于希腊语的 舵手、飞行员。在国内又称k8s&#xff08;因为k和s之间有8个字母&#xff0c;所以得名。“国内程序员的幽默”&#xff09;。 作用&#xff1a; 用于自动部署、扩展和管理“容器化&#x…

Python 小抄

Python 备忘单 目录 1.语法和空格 2.注释 3.数字和运算 4.字符串处理 5.列表、元组和字典 6.JSON 7.循环 8.文件处理 9.函数 10.处理日期时间 11.NumPy 12.Pandas 要运行单元格&#xff0c;请按 ShiftEnter 或单击页面顶部的 Run&#xff08;运行&#xff09;。 1.语法和空格…

企业微信网页应用开发(java版)

1、成为开发者 企业微信先认证企业,成为开发者,到开发者中心创建web应用 开发者中心 选择工具 2、创建一个web应用 填写web应用信息 填写配置开发信息 域名地址先填一样的,填好后创建应用 3、配置可信域名 点击检查可信域名归属 下载可信域名验证文件 将下好的文件放到…

资产系统可视化后台 vue2+antd

步骤条 <template><div style"margin-left: 30px"><!-- <a-card title"审批进度" style"margin-top: 10px"><a-table size"default" :columns"columns" :data-source"schedule" bordere…

VUE H5字体在安卓手机偏上解决

安卓手机展示样式,数字偏上,展示效果如图: 项目内添加新字体,引用新字体 vue 项目需要引入字体的话, 可以移步到这篇文章(无需下载依赖包)Vue3中引入外部自定义字体 项目文件assets内创建font文件夹, 粘贴你想用的字体, 创建对应的css文件; scss代码: font-face {/* 自定义的…