随机增量构造


一,随机增量构造

概述:在解决前 n − 1 n-1 n1 规模的问题前提之下,使得第 n n n 个新加入的元素也是符合条件的
注意:结合随机化,积累构造法则
例子:扩展KMP,最小圆覆盖

二,最小圆覆盖

  • 要素:三层循环

引理: 证明在这里

1,最小圆覆盖在确定其中点的情况下唯一确认
2,若是第 i i i 个点不在前面点的最小圆覆盖上,那么一定在可能的圆的边界上

步骤:

  1. 随机化点集
  2. 初始时一个点的最小覆盖圆就是这个点本身
  3. 假设已经求出前 i − 1 i−1 i1个点的最小覆盖圆
  4. 若第 i i i 个点在圆内,则跳过;
  5. 若第 i i i 个点不在圆内,根据性质2,只需要找出一个圆覆盖前 i − 1 i−1 i1个点,且第 i i i个点在圆边上
  6. 由于三点确定一个圆,继续找第二个点j,类似于性质2,现在需要找出一个圆覆盖前 j − 1 j−1 j1个点,且第 i i i和第 j j j个点在圆上
  7. 最后找第三个点 k k k,同样现在需要找出一个圆覆盖前 k − 1 k−1 k1个点,且第 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);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

流苏贺风

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值