\(\color{#0066ff}{题目描述}\)

给出N个点,让你画一个最小的包含所有点的圆。

\(\color{#0066ff}{输入格式}\)

先给出点的个数N,2<=N<=100000,再给出坐标Xi,Yi.(-10000.0<=xi,yi<=10000.0)

\(\color{#0066ff}{输出格式}\)

输出圆的半径,及圆心的坐标,保留10位小数

\(\color{#0066ff}{输入样例}\)

  1. 6
  2. 8.0 9.0
  3. 4.0 7.5
  4. 1.0 2.0
  5. 5.1 8.7
  6. 9.0 2.0
  7. 4.5 1.0

\(\color{#0066ff}{输出样例}\)

  1. 5.0000000000
  2. 5.0000000000 5.0000000000

\(\color{#0066ff}{数据范围与提示}\)

none

\(\color{#0066ff}{题解}\)

随机增量法

前置知识,三点定圆

设圆上有三个点\(A(x_1, y_1), B(x_2, y_2), C(x_3, y_3)\)

我们可以找这些点的连线的任两条线,找它们垂直平分线的交点就是圆心

易得向量(线段)AB,BC

设两条线段的垂直平分线为\(A_1x+B_1y=C_1,A_2x+B_2y=C_2\)

上式的A和B分别是两个向量的x和y(法向量(可以带入验证))

之后带入AB,BC的中点来求出两个C

联立两个垂直平分线,解出来

\(x=\frac{C_2*B_1-C_1*B_2}{A_2*B_1-A_1*B_2}, y=\frac{C_2*A_1-C_1*A_2}{B_2*A_1-B_1*A_2}\)

有了圆心,半径自然好求,随便找圆上一点(A,B,C)与圆心求距离就是半径了

随机增量

初始设置第一个点为圆心,半径为0

依次扫每个点,如果不在当前圆内,则以那个点为圆心

再次扫从头开始到当前的每个点,如果不在当前圆中

则当前点与那个点作为新圆直径,构成当前圆,再次扫到当前每个点

如果仍有不在圆内的,三点定圆

  1. #include <bits/stdc++.h>
  2. #define _ 0
  3. #define LL long long
  4. inline LL in() {
  5. LL x = 0, f = 1; char ch;
  6. while(!isdigit(ch = getchar()))(ch == '-') && (f = -f);
  7. while(isdigit(ch)) x = x * 10 + (ch ^ 48), ch = getchar();
  8. return x * f;
  9. }
  10. const int maxn = 1e5 + 10;
  11. const double eps = 1e-14;
  12. struct node {
  13. double x, y;
  14. node(double x = 0, double y = 0)
  15. :x(x), y(y) {}
  16. double mod() {
  17. return sqrt(x * x + y * y);
  18. }
  19. friend node operator - (const node &a, const node &b) {
  20. return node(a.x - b.x, a.y - b.y);
  21. }
  22. friend node operator + (const node &a, const node &b) {
  23. return node((a.x + b.x) / 2.0, (a.y + b.y) / 2.0);
  24. }
  25. }e[maxn];
  26. int n;
  27. void circle(node a, node b, node c, node &o, double &r) {
  28. node ab = b - a, bc = c - b, mid1 = a + b, mid2 = b + c;
  29. double A1 = ab.x, B1 = ab.y, A2 = bc.x, B2 = bc.y;
  30. double C1 = A1 * mid1.x + B1 * mid1.y, C2 = A2 * mid2.x + B2 * mid2.y;
  31. o = node((C2 * B1 - C1 * B2) / (A2 * B1 - A1 * B2), (C2 * A1 - C1 * A2) / (B2 * A1 - B1 * A2));r = (o - a).mod();
  32. }
  33. void work() {
  34. node o = e[1];
  35. double r = 0;
  36. for(int i = 2; i <= n; i++)
  37. if((e[i] - o).mod() - r > eps) {
  38. o = e[i], r = 0;
  39. for(int j = 1; j <= i - 1; j++)
  40. if((e[j] - o).mod() - r > eps) {
  41. o = e[i] + e[j];
  42. r = (o - e[i]).mod();
  43. for(int k = 1; k <= j - 1; k++)
  44. if((e[k] - o).mod() - r > eps)
  45. circle(e[i], e[j], e[k], o, r);
  46. }
  47. }
  48. printf("%.10f\n%.10f %.10f", r, o.x, o.y);
  49. }
  50. int main() {
  51. n = in();
  52. for(int i = 1; i <= n; i++) scanf("%lf%lf", &e[i].x, &e[i].y);
  53. work();
  54. return 0;
  55. }

P1742 最小圆覆盖的更多相关文章

  1. 洛谷 P1742 最小圆覆盖 (随机增量)

    题目链接:P1742 最小圆覆盖 题意 给出 N 个点,求最小的包含所有点的圆. 思路 随机增量 最小圆覆盖一般有两种做法:随机增量和模拟退火.随机增量的精确度更高,这里介绍随机增量的做法. 先将所有 ...

  2. 洛谷P1742 最小圆覆盖(计算几何)

    题面 传送门 题解 之前只是在抄题解--这篇才算是真正自己想的吧-- 首先我们把输入序列给\(random\)一下防止出题人好心送你一个毒瘤序列 我们设\(r\)为当前最大半径,\(o\)为此时对应圆 ...

  3. (bzoj1337 || 洛谷P1742 最小圆覆盖 )|| (bzoj2823 || 洛谷P2533 [AHOI2012]信号塔)

    bzoj1337 洛谷P1742 用随机增量法.讲解:https://blog.csdn.net/jokerwyt/article/details/79221345 设点集A的最小覆盖圆为g(A) 可 ...

  4. P1742 最小圆覆盖(计算几何)

    体验过\(O(n^3)\)过\(10^5\)吗?快来体验一波当\(wys\)的快感吧\(QAQ\) 前置芝士1:二元一次方程组求解 设 \[\begin{cases}a1 * x + b1*y=c1\ ...

  5. 洛谷P1742 最小圆覆盖(计算几何)

    题意 题目链接 Sol 暴力做法是\(O(n^3)\)枚举三个点然后check一下是否能包含所有点 考虑一种随机算法,首先把序列random_shuffle一下. 然后我们枚举一个点\(i\),并维护 ...

  6. 最小圆覆盖(洛谷 P1742 增量法)

    题意:给定N个点,求最小圆覆盖的圆心喝半径.保留10位小数点. N<1e5: 思路:因为精度要求较高,而且N比较大,所以三分套三分的复杂度耶比较高,而且容易出错. 然是写下增量法吧. 伪代码加深 ...

  7. 【BZOJ-1336&1337】Alie最小圆覆盖 最小圆覆盖(随机增量法)

    1336: [Balkan2002]Alien最小圆覆盖 Time Limit: 1 Sec  Memory Limit: 162 MBSec  Special JudgeSubmit: 1573   ...

  8. Bzoj 1336&1337 Alien最小圆覆盖

    1336: [Balkan2002]Alien最小圆覆盖 Time Limit: 1 Sec  Memory Limit: 162 MBSec  Special Judge Submit: 1473  ...

  9. hdu3007Buried memory(最小圆覆盖)

    链接 普通的暴力复杂度达到O(n^4),对于这题肯定是不行的. 解法:随机增量算法 参考http://www.2cto.com/kf/201208/149602.html algorithm:A.令C ...

随机推荐

  1. QQ控件时光轴特效总结

    1.插入HTML数据 插入html代码,一般的做法是通过document.getElementById("").innerHTML来实现. 然而在该控件中,它通过JS replac ...

  2. 转:InnoDB多版本(MVCC)实现简要分析

    InnoDB多版本(MVCC)实现简要分析 基本知识 假设对于多版本(MVCC)的基础知识,有所了解.InnoDB为了实现多版本的一致读,采用的是基于回滚段的协议. 行结构 InnoDB表数据的组织方 ...

  3. 2015.1.15 利用航线id取所有点的函数创建视图

    1.根据航路id取所有航路点的函数 create or replace function alinepnts(alid in number) return tab_airline_pnt is --返 ...

  4. node install error

    错误:Unexpected end of JSON input while parsing near.... 解决办法: npm cache clean --force

  5. leetcode633

    用开方的思想来解题. bool judgeSquareSum(int c) { int h = pow(c, 0.5); ; i <= h; i++) { ), 0.5); if (left - ...

  6. sql 一些偶尔会用到的写法和函数 不定时更新

    小数转整数: --round() 遵循四舍五入把原值转化为指定小数位数,如: ) -- =1 ) -- =2 --floor() 向下舍入为指定小数位数 如: SELECT floor(1.45) - ...

  7. IB使用

    A:给控件添加方法或变量. 1.窗口上拖个控件 NSButton 2..点右上那张狗脸(Editor)对上的. 3.右键控件.拖到头文件中. 4 .选择加响应方法或变量.

  8. .net中值类型、引用类型理解的c#代码示例

    下面是以前在公司的时候给别人讲解值类型.引用类型时创建的c#代码示例,从实际使用时的角度出发,对于初学者还是很有帮助的.这里并没有深入讲解值类型包含引用类型成员时(如struct)在内存中的存放情况等 ...

  9. C# 与Java初始化顺序及异同(转)

    C#初始化顺序 类成员变量初始化先于类的构造函数 静态成员变量先于实例变量 子类成员变量先于父类成员变量(java相反) 父类构造函数先于子类构造函数 参考实例: using System; //us ...

  10. ubuntu 删除mysql

    UBUNTU 彻底删除 MYSQL 然后重装 MYSQL 删除 mysql sudo apt-get autoremove --purge mysql-server-5.0sudo apt-get r ...