bzoj 2458: [BeiJing2011]最小三角形 题解

【前言】话说好久没有写题解了。到暑假了反而忙。o(╯□╰)o

【原题】

2458: [BeiJing2011]最小三角形

Time Limit: 10 Sec  Memory Limit: 128 MB

Submit: 574  Solved: 177

[Submit][Status]

Description

Xaviera如今遇到了一个有趣的问题。

平面上有N个点。Xaviera想找出周长最小的三角形。

因为点许多。分布也很乱,所以Xaviera想请你来解决问题。

为了减小问题的难度,这里的三角形也包含共线的三点。

Input

第一行包括一个整数N表示点的个数。

接下来N行每行有两个整数,表示这个点的坐标。

Output

输出仅仅有一行。包括一个6位小数,为周长最短的三角形的周长(四舍五入)。

Sample Input

4

1 1

2 3

3 3

3 4

Sample Output

3.414214

HINT

100%的数据中N≤200000。

Source

Day1

【分析】今天新学了解决这类问题的方法——分治。

没错,就是分治。

先讲一下n是10^5级别的平面近期点对吧(CF 245 DIV 2 D)。非常easy懂。

具体的原理能够參考这个博客。讲的非常具体。

(非常多时候仅仅要感性认识原理就可以)以下讲一下具体做法。

①对于平面上的点。按x坐标排序(这是永久排序)。

②每次递归(l。r)。函数的返回值是第l个到第r个之间的全部点的近期点对。

③假设l=r。那么返回无穷大;假设l+1=r。就直接返回两个点的距离。

④每次先递归(l。mid)和(mid+1,r)。显然,这两个会有两个返回值。最好还是设为d1和d2。

首先我们设D=MIN(d1,d2)。即当前的最优值临时是D。

⑤显然,另一种情况。左边那块的某个点和右边那块的某个点产生关系。那么,我们能够从mid这个位置向左跑到mid-D,向右跑到mid+D,然后把这一段中的点都拎出来——由于仅仅有这两段中的点才有可能产生小于D的贡献。

⑥这时候我们要意识到潜在复杂度的保证(实际原理也不难懂呵)。

首先,假设直接枚举两两点要N^2。我们先把拎出来的点按y排序。

(NlogN)然后看似也是N^2的枚举,仅仅是加了一个优化(从底下開始枚举i。假设Y[J]-Y[I]>D就直接break)——这样可证明差点儿是线性。

总复杂度N*logN*logN。

再讲一下本题。也是几乎相同道理。

由于是三角形,我们把一些细节改一下就可以。

【代码】

#include<cstdio>
#include<cmath>
#include<algorithm>
#define N 200005
#define INF 210000000000.0
using namespace std;
struct arr{int x,y;}a[N],num[N];int n,i,Test;
inline bool cmpx(const arr &a,const arr &b){return a.x<b.x;}
inline bool cmpy(const arr &a,const arr &b){return a.y<b.y;}
inline double dis(const arr &a,const arr &b){return sqrt((a.x-b.x)*1.*(a.x-b.x)+(a.y-b.y)*1.*(a.y-b.y));}
inline double work(int l,int r)
{
  if (l==r) return INF;
  if (l+1==r) return INF;
  if (l+2==r) return dis(a[l],a[l+1])+dis(a[l+1],a[r])+dis(a[l],a[r]);
  int mid=(l+r)>>1;
  double d1=work(l,mid),d2=work(mid+1,r);
  double D=min(d1,d2),ans=D,DD=D/2.0;int cnt=0;
  for (int i=l;i<=r;i++)
    if (fabs(a[mid].x-a[i].x)<=DD) num[++cnt]=a[i];
  sort(num+1,num+cnt+1,cmpy);
  for (int i=1;i<cnt-1;i++)
    for (int j=i+1;j<cnt;j++)
    {
      if (num[j].y-num[i].y>DD) break;
      for (int k=j+1;k<=cnt;k++)
      {
        if (num[k].y-num[i].y>DD) break;
        double temp=dis(num[i],num[j])+dis(num[i],num[k])+dis(num[j],num[k]);
        if (temp<ans) ans=temp;
      }
    }
  return ans;
}
int main()
{
  read(n);//读入优化就不贴了。

for (i=1;i<=n;i++)
    read(a[i].x),read(a[i].y);
  sort(a+1,a+n+1,cmpx);
  printf("%.6lf",work(1,n));
  return 0;
}
时间: 2025-01-04 21:07:49

bzoj 2458: [BeiJing2011]最小三角形 题解的相关文章

BZOJ 2458 BeiJing2011 最小三角形 计算几何+分治

题目大意:给定平面上的一个点集,求这个点集所能组成的周长最小的三角形 与平面最近点对一个道理- - 这个题也是分治做法 做法如下: 1.记录全局答案ans 2.将所有点按照x值排序 3.定义Solve(l,r)为处理[l,r]区间内的最小三角形 4.对于每层Solve(l,r),将当前区间分成左右两部分,分别递归处理 5.两侧的最小三角形都以处理完毕,现在我们要处理的就是两区间之间的点构成的三角形 6.将本层中与点mid的横坐标之差不超过ans/2的点拎出来,按照纵坐标排序 (其实这步可以直接递

BZOJ 2458: [BeiJing2011]最小三角形 | 平面分治

题目: 给出若干个点 求三个点构成的周长最小的三角形的周长(我们认为共线的三点也算三角形) 题解: 可以参考平面最近点对的做法 只不过合并的时候改成枚举三个点更新周长最小值,其他的和最近点对大同小异 2#include<cstdio> #include<algorithm> #include<cstring> #include<cmath> #define N 200010 #define INF 1e20 using namespace std; int

bzoj2458: [BeiJing2011]最小三角形(分治+几何)

题目链接:bzoj2458: [BeiJing2011]最小三角形 学习推荐博客:分治法编程问题之最接近点对问题的算法分析 题解:先将所有点按x值排列,然后每次将当前区间[l,r]分成左右两半递归求解周长最小三角形.考虑到两半区间之间可能有连成最小三角形的情况,设dd为两半区间中最小三角形周长的最小值,筛选满足要求的点(x值与中点坐标x值的距离小于dd),然后按y值排序,进而暴搜出周长最小三角形. 1 #include<cstdio> 2 #include<cmath> 3 #in

BZOJ2458 [BeiJing2011]最小三角形

Description Xaviera现在遇到了一个有趣的问题. 平面上有N个点,Xaviera想找出周长最小的三角形. 由于点非常多,分布也非常乱,所以Xaviera想请你来解决这个问题. 为了减小问题的难度,这里的三角形也包括共线的三点. Input 第一行包含一个整数N表示点的个数. 接下来N行每行有两个整数,表示这个点的坐标. Output 输出只有一行,包含一个6位小数,为周长最短的三角形的周长(四舍五入). Sample Input 4 1 1 2 3 3 3 3 4 Sample

bzoj2458: [BeiJing2011]最小三角形 最近点对

类似分治最近点对的方法乱搞一下就行. #include<bits/stdc++.h> #define N 200010 #define M (s+t>>1) using namespace std; struct vec{ int x,y; }u[N],v[N]; bool foo(vec a,vec b){ return a.x<b.x; } bool bar(vec a,vec b){ return a.y<b.y; } double sqr(double x){

BZOJ 2458 最小三角形 | 平面分治

BZOJ 2458 最小三角形 题面 一个平面上有很多点,求他们中的点组成的周长最小的三角形的周长. 题解 跟平面最近点对差不多,也是先把区间内的点按x坐标从中间分开,递归处理,然后再处理横跨中线的三角形. 如何缩小范围?设左右两个子区间发现的最小周长是d,则与中线距离超过d / 2都没有用了,对于一个点,所有与它距离超过d / 2的点也都没有用. #include <cstdio> #include <cstring> #include <cmath> #includ

bzoj 1874 取石子游戏 题解 &amp; SG函数初探

[原题] 1874: [BeiJing2009 WinterCamp]取石子游戏 Time Limit: 5 Sec  Memory Limit: 162 MB Submit: 334  Solved: 122 [Submit][Status] Description 小H和小Z正在玩一个取石子游戏. 取石子游戏的规则是这样的,每个人每次可以从一堆石子中取出若干个石子,每次取石子的个数有限制,谁不能取石子时就会输掉游戏. 小H先进行操作,他想问你他是否有必胜策略,如果有,第一步如何取石子. In

斜率优化专题5——bzoj 1096 [ZJOI2007]仓库建设 题解

[原题] 1096: [ZJOI2007]仓库建设 Time Limit: 10 Sec  Memory Limit: 162 MB Submit: 1998  Solved: 816 [Submit][Status] Description L公司有N个工厂,由高到底分布在一座山上.如图所示,工厂1在山顶,工厂N在山脚. 由于这座山处于高原内陆地区(干燥少雨),L公司一般把产品直接堆放在露天,以节省费用.突然有一天,L公司的总裁L先生接到气象部门的电话,被告知三天之后将有一场暴雨,于是L先生决

【模板】【计几】旋转坐标(用于最小三角形以及三角形四边形面积存在性问题)

发现计算几何算法(瞎搞)真的是博大精深. 最大三角形和最大四边形都是旋转卡壳,有模板的.这里的方法还可以求最小三角形还有最小四边形,以及三角形面积存在性问题. 求最小三角形面积(n平方):bzoj3707. 参考:http://www.pianshen.com/article/772191644/ 其实就是先把n方个直线按照斜率先排了序,然后所有点按照距离y轴距离排序(距离有正负),然后枚举的直线相当于y轴,每一次旋转 “ y轴 ” 的时候维护一下这个序列.然后我们发现维护这个序列只需要交换现在