随机增量构造
- 一,随机增量构造
- 二,最小圆覆盖
一,随机增量构造
概述:在解决前 n−1n-1n−1 规模的问题前提之下,使得第nnn 个新加入的元素也是符合条件的
注意:结合随机化,积累构造法则
例子:扩展KMP,最小圆覆盖
二,最小圆覆盖
引理: 证明在这里
1,最小圆覆盖在确定其中点的情况下唯一确认
2,若是第 iii 个点不在前面点的最小圆覆盖上,那么一定在可能的圆的边界上
步骤:
- 随机化点集
- 初始时一个点的最小覆盖圆就是这个点本身
- 假设已经求出前i−1i−1i−1个点的最小覆盖圆
- 若第 iii 个点在圆内,则跳过;
- 若第 iii 个点不在圆内,根据性质2,只需要找出一个圆覆盖前i−1i−1i−1个点,且第iii个点在圆边上
- 由于三点确定一个圆,继续找第二个点j,类似于性质2,现在需要找出一个圆覆盖前j−1j−1j−1个点,且第iii和第jjj个点在圆上
- 最后找第三个点kkk,同样现在需要找出一个圆覆盖前k−1k−1k−1个点,且第iii和第jjj和第kkk个点在圆边上(那么说明,只要发现不在里面,那么圆就得适应新的点,保证前面的已经实现,所以不需要考虑条件限制)
pair<pdd,pdd> get_line(pdd a,pdd b)
{return {(a&#43;b)/2,spin(b-a,PI/2)};
}node get_node(pdd a,pdd b,pdd c)
{auto u &#61; get_line (a,b);auto v &#61; get_line (a,c);auto p &#61; get_line_join(u.x,u.y,v.x,v.y);return {p,get_dist(p,a)};
}int main()
{scanf("%d", &n);for (int i &#61; 0; i < n; i &#43;&#43; ) scanf("%lf%lf", &q[i].x, &q[i].y);random_shuffle(q, q &#43; n);node c &#61; {q[0],0};for(int i&#61;1;i<n;i&#43;&#43;){if(dcmp(c.r,get_dist(c.p,q[i]))<0){c &#61; {q[i],0};for(int j&#61;0;j<i;j&#43;&#43;){if(dcmp(c.r,get_dist(c.p,q[j]))<0){c &#61; {(q[i] &#43; q[j]) / 2, get_dist(q[i], q[j]) / 2};for(int k&#61; 0;k<j;k&#43;&#43;){if(dcmp(c.r,get_dist(c.p,q[k]))<0)c &#61; get_node(q[i],q[j],q[k]);}}}}}printf("%.10lf\n", c.r);printf("%.10lf %.10lf\n", c.p.x, c.p.y);
}