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