传送门

这整场都不会……这题想二分不会check

其实check很好写,考虑一个mid的实际意义

即为check在不靠近每个star及边界mid距离内的前提下,能不能到达\((n,m)\)

其实可以转化一下,以每个star为圆心,mid为半径画圆

如果两个圆相交就在这两个圆之间连边

并查集维护连通性,如果上下边界连通就说明所有可行路径都被封上了

可以\(O(k^2logn)\)二分

但是还有\(O(k^2)\)解法:

考虑是什么在限制整条路径上的最小值

如果将所有star之间连边,就一定有几条权值最小的边连出了一条从上边界到下边界的链

无论如何都必须通过这条链

可以构建一棵最小生成树,就确保了权值最小

那问题可以转化为在权值最小的这条链上找一条权值最大的边

  • 解最小值最大问

prim板子:

我承认当年因为觉得kruskal复杂度更优我prim是水过去的,但事实上完全图上kruskal多个log

  1. void prim() {
  2. memset(d, 127, sizeof(d));
  3. d[1]=0;
  4. for (int i=1,x; i<n; ++i) {
  5. x=0;
  6. for (int j=1; j<=n; ++j)
  7. if (!vis[j] && (!x||d[j]<d[x])) x=j;
  8. vis[x]=1;
  9. for (int j=1; j<=n; ++j) if (!vis[j]) d[j]=min(d[j], dis[x][j]);
  10. }
  11. }

其它细节见代码注释

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. #define INF 0x3f3f3f3f
  4. #define N 6005
  5. #define ll long long
  6. #define ld long double
  7. #define usd unsigned
  8. #define ull unsigned long long
  9. //#define int long long
  10. #define getchar() (p1==p2&&(p2=(p1=buf)+fread(buf, 1, 1<<21, stdin)), p1==p2?EOF:*p1++)
  11. char buf[1<<21], *p1=buf, *p2=buf;
  12. inline int read() {
  13. int ans=0, f=1; char c=getchar();
  14. while (!isdigit(c)) {if (c=='-') f=-f; c=getchar();}
  15. while (isdigit(c)) {ans=(ans<<3)+(ans<<1)+(c^48); c=getchar();}
  16. return ans*f;
  17. }
  18. int n, m, k;
  19. double x[N], y[N];
  20. namespace force{
  21. double ans=1e30;
  22. void solve() {
  23. for (int i=1; i<=k; ++i) {
  24. ans=min(ans, min(y[i], 1.0*m-y[i]));
  25. for (int j=i+1; j<=k; ++j) ans=min(ans, sqrt((x[i]-x[j])*(x[i]-x[j])+(y[i]-y[j])*(y[i]-y[j])));
  26. }
  27. printf("%.8lf\n", ans);
  28. exit(0);
  29. }
  30. }
  31. namespace task_MLE{
  32. double dis[N][N], d[N];
  33. short fa[N];
  34. bool vis[N];
  35. inline int find(int p) {return fa[p]==p?p:fa[p]=find(fa[p]);}
  36. void solve() {
  37. //cout<<double(sizeof(dis))/1024/1024<<endl;
  38. memset(d, 0x7f, sizeof(d));
  39. memset(dis, 0x7f, sizeof(dis));
  40. for (int i=1; i<=k; ++i)
  41. for (int j=1; j<=k; ++j)
  42. if (i!=j) dis[i][j]=dis[j][i]=sqrt((x[i]-x[j])*(x[i]-x[j])+(y[i]-y[j])*(y[i]-y[j])); //, cout<<"dis"<<i<<' '<<j<<' '<<dis[i][j]<<endl;
  43. for (int i=1; i<=k; ++i) dis[i][k+1]=dis[k+1][i]=y[i];
  44. for (int i=1; i<=k; ++i) dis[i][k+2]=dis[k+2][i]=1.0*m-y[i];
  45. for (int i=1; i<=k+2; ++i) fa[i]=i;
  46. d[1]=0;
  47. for (int i=1,x; i<=k+1; ++i) {
  48. x=0;
  49. for (int j=1; j<=k+2; ++j)
  50. if (!vis[j] && (!x||d[j]<d[x])) x=j;
  51. vis[x]=1; fa[find(x)]=find(1);
  52. //cout<<"x: "<<x<<endl;
  53. for (int j=1; j<=k+2; ++j) if (!vis[j]) d[j]=min(d[j], dis[x][j]); //, cout<<"d["<<j<<"]: "<<d[j]<<endl;
  54. if (find(k+1)==find(k+2)) break;
  55. }
  56. double maxn=0;
  57. for (int i=1; i<=k+2; ++i) if (vis[i]) maxn=max(maxn, d[i]);
  58. printf("%.8lf\n", maxn/2);
  59. }
  60. }
  61. namespace task{
  62. double d[N], ans;
  63. short fa[N];
  64. bool vis[N];
  65. inline int find(int p) {return fa[p]==p?p:fa[p]=find(fa[p]);}
  66. inline double dis(int i, int j) {
  67. if (i>k||j>k) {
  68. if (i>k&&j>k) return 1e30;
  69. if (i<=k) swap(i, j);
  70. return (i==k+1)?y[j]:(1.0*m-y[j]);
  71. }
  72. else return sqrt((x[i]-x[j])*(x[i]-x[j])+(y[i]-y[j])*(y[i]-y[j]));
  73. }
  74. void solve() {
  75. memset(d, 0x7f, sizeof(d));
  76. for (int i=1; i<=k+2; ++i) fa[i]=i;
  77. d[k+1]=0;
  78. int cnt=0;
  79. for (int i=1,x; i<=k+1; ++i) {
  80. cnt=i;
  81. x=0;
  82. for (int j=1; j<=k+2; ++j)
  83. if (!vis[j] && (!x||d[j]<d[x])) x=j;
  84. vis[x]=1; fa[find(x)]=find(k+1);
  85. //cout<<x<<endl;
  86. // 这里一开始的写法是因为没有注意到只有从k+1开始搜,才能满足所搜到的所有能对答案产生贡献的点都在连接上下边界的最小路径上
  87. // 由于prim的性质,我们所加到集合中的点不一定会在那条最小路径上,但考虑最后当我们加入一个点,k+1和k+2变得连通时
  88. // 这个点一定在最小路径上,而之前加的点分两种情况
  89. // 若不在最小路径上,则由它扩展出来的点也不在最小路径上,因为我们要求最大值,最后一定需要加入权值更大的边使图连通,所以它一定不对答案产生贡献
  90. // 若在最小路径上,注意一个事情(这里被卡了), 因为prim维护的是点集,而且是基于连通性扩展的,所以可以构造出一种情况,
  91. // 使后加的边权值更小(之前它因为和已知集合中的点没有连边而没有被选进去),所以**prim所扩展的边权值不一定递增**
  92. for (int j=1; j<=k+2; ++j) if (!vis[j]) d[j]=min(d[j], dis(x, j)); //, cout<<"d["<<j<<"]: "<<d[j]<<endl;
  93. if (find(k+1)==find(k+2) || i==k+1) {
  94. double maxn=0;
  95. for (int i=1; i<=k+2; ++i) if (vis[i]) maxn=max(maxn, d[i]);
  96. printf("%.8lf\n", maxn/2);
  97. //printf("%.8lf\n", d[x]/2);
  98. exit(0);
  99. }
  100. }
  101. }
  102. }
  103. signed main()
  104. {
  105. #ifdef DEBUG
  106. freopen("1.in", "r", stdin);
  107. #endif
  108. n=read(); m=read(); k=read();
  109. for (int i=1; i<=k; ++i) x[i]=read(), y[i]=read();
  110. //if (k==1) {printf("%.8lf", max(y[1], m-y[1])/2); return 0;}
  111. //if (k==2) {printf("%.8lf", min(min(max(y[1], m-y[1]), max(y[2], m-y[2])), sqrt((x[1]-x[2])*(x[1]-x[2])+(y[1]-y[2])*(y[1]-y[2])))/2); return 0;}
  112. //force::solve();
  113. task::solve();
  114. return 0;
  115. }

题解 Star Way To Heaven的更多相关文章

  1. [CSP-S模拟测试]:Star Way To Heaven(最小生成树Prim)

    题目描述 小$w$伤心的走上了$Star\ way\ to\ heaven$. 到天堂的道路是一个笛卡尔坐标系上一个$n\times m$的长方形通道(顶点在$(0,0)$和$(n,m)$),小$w$ ...

  2. 7.15考试总结(NOIP模拟16)[Star Way To Heaven·God Knows·Lost My Music]

    败者死于绝望,胜者死于渴望. 前言 一看这个题就来者不善,对于第一题第一眼以为是一个大模拟,没想到是最小生成树. 对于第二题,先是看到了状压可以搞到的 20pts 然后对着暴力一顿猛调后来发现是题面理 ...

  3. NOIP模拟16:「Star Way To Heaven·God Knows·Loost My Music」

    T1:Star Way To Heaven 基本思路:   最小生成树.   假如我们将上边界与下边界看作一个点,然后从上边界经过星星向下边界连边,会发现,他会形成一条线将整个矩形分为左右两个部分. ...

  4. Star Way To Heaven

    题目描述 小 x伤心的走上了 Star way to heaven. 到天堂的道路是一个笛卡尔坐标系上一个 n*m的长方形通道 顶点在0,0 和 . 小 n,m 从最左边任意一点进入,从右边任意一点走 ...

  5. 20190817-T1-LOJ6322「雅礼国庆 2017 Day6」Star Way To Heaven

    写这篇题解是因为作者太蒻已经忘了最小生成树了. <题面> 这个题还真是想不到最小生成树. $80\%$算法 复杂度:$\Theta(k^2 \log N )$ 用了二分答案(明显答案具有单 ...

  6. NOIP 模拟 $16\; \rm Star Way To Heaven$

    题解 \(by\;zj\varphi\) 看懂题!!! 从最左穿到最右,一定会经过两个星星之间或星星和边界之间,那么我们穿过时当前最优一定是走中点 而我们要求最小的距离最大,那么我们将所有星星和边界( ...

  7. 「模拟8.17」star way to heaven(并查集,最小生成树)

    80分打法 首先二分最后答案,答案即为r,可看作以每个k为圆心r为半径的圆 我们进行并查集维护,维护相交的圆的边界 最后判断是否存在圆将上下边界覆盖,如有证明不行 1 #include<iost ...

  8. NOIP模拟测试24「star way to hevaen·lost my music」

    star way to heaven 题解 大致尝试了一下并查集,记忆化搜索,最小生成树 最小生成树是正解,跑最小生成树然后找到最大的值 欧几里德距离最小生成树学习 prim楞跑 至于为什么跑最小生成 ...

  9. NOIp2018集训test-9-17(am)

    这是一套去年在长沙考过的题,但是我当时就没理清楚+没写题解(我以前很多博客都写得跟shi一样,完全没有意义,看到就想打当时的我),所以又考得稀烂. T1.star way to heaven 容易想到 ...

随机推荐

  1. 『心善渊』Selenium3.0基础 — 24、Selenium的expected_conditions模块详细介绍

    目录 1.EC模块介绍 2.EC模块常用类 3.EC模块的使用 4.EC模块综合使用 (1)title_is(title)示例 (2)presence_of_element_located(locat ...

  2. ESP32存储blog笔记

    基于ESP-IDF4.1 1 #include <stdio.h> 2 #include "freertos/FreeRTOS.h" 3 #include " ...

  3. fastjson: json对象,json对象数组,javabean对象,json字符串之间的相互转化

    fastjson: json对象,json对象数组,javabean对象,json字符串之间的相互转化 在开发过程中,经常需要和前端交互数据,数据交互的格式都是JSON,在此过程中免不了json字符串 ...

  4. ACM金牌选手算法讲解《线性表》

    哈喽,大家好,我是编程熊,双非逆袭选手,字节跳动.旷视科技前员工,ACM亚洲区域赛金牌,保研985研究生,分享算法与数据结构.计算机学习经验,帮助大家进大厂~ 公众号:『编程熊』 文章首发于: ACM ...

  5. FTP传输

    FTP传输                    一.FTP服务–用来传输文件的协议                    二.设置匿名用户访问的FTP服务(最大权限)                 ...

  6. python 得到字典的所有键 和值

    a={} a={"a":1,"b":2,"c":3,"d":4} print(a) print(a.items()) p ...

  7. [刘阳Java]_酷炫视频播放器制作_JS篇

    此文章是接着上次写的<酷炫视频播放器制作_界面篇>将其完善,我们主要给大家介绍一下如何利用JS脚本来控制视频的播放.为了让大家能够保持对要完成的功能有直接的了解,我们还是将效果图附到文章里 ...

  8. 学习总结 NCRE二级和三级

    NCRE二级C语言 证书 考试感想 2016年考的认证,5年过去了,"光阴荏苒真容易".趁着心有余力有余的时候,把一些个人的体会分享给大家,希望后来人能平稳前行. Windows ...

  9. Maven多模块Spring的注入

    第一次做多模块项目,Unit Test的时候发现Biz模块没法注入Dal模块的Mappper,提示找不到Bean. POM文件是正确引入了dependency的,扫描的package也没错,所以卡了好 ...

  10. odoo里API解读

    Odoo自带的api装饰器主要有:model,multi,one,constrains,depends,onchange,returns 七个装饰器. multimulti则指self是多个记录的合集 ...