bzoj 2458: [BeiJing2011]最小三角形 题解
【前言】话说好久没有写题解了。到暑假了反而忙。o(╯□╰)o
【原题】
2458: [BeiJing2011]最小三角形
Time Limit: 10 Sec Memory Limit: 128 MB id=2458" style="color:blue; text-decoration:none">Submit
Submit: 574 Solved: 177
[
Description
Xaviera如今遇到了一个有趣的问题。
平面上有N个点。Xaviera想找出周长最小的三角形。
因为点许多。分布也很乱,所以Xaviera想请你来解决问题。
为了减小问题的难度,这里的三角形也包含共线的三点。
Input
第一行包括一个整数N表示点的个数。
接下来N行每行有两个整数,表示这个点的坐标。
Output
输出仅仅有一行。包括一个6位小数,为周长最短的三角形的周长(四舍五入)。
Sample Input
1 1
2 3
3 3
3 4
Sample Output
HINT
100%的数据中N≤200000。
Source
【分析】今天新学了解决这类问题的方法——分治。
没错,就是分治。
先讲一下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;
}
bzoj 2458: [BeiJing2011]最小三角形 题解的更多相关文章
- BZOJ 2458: [BeiJing2011]最小三角形 | 平面分治
题目: 给出若干个点 求三个点构成的周长最小的三角形的周长(我们认为共线的三点也算三角形) 题解: 可以参考平面最近点对的做法 只不过合并的时候改成枚举三个点更新周长最小值,其他的和最近点对大同小异 ...
- [BZOJ]2458: [BeiJing2011]最小三角形
题目大意:给出平面上n个点,求最小的由这些点组成的三角形的周长.(N<=200,000) 思路:点按x坐标排序后分治,每次取出与排在中间的点的横坐标相差不超当前答案一半的点,按y坐标排序后再暴力 ...
- BZOJ 2458: [BeiJing2011]最小三角形 (分治)
分治就是了. 类似于分治找最近/远点对. CODE #include <bits/stdc++.h> using namespace std; const double eps = 1e- ...
- bzoj-2458 2458: [BeiJing2011]最小三角形(计算几何+分治)
题目链接: 2458: [BeiJing2011]最小三角形 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1101 Solved: 380 Des ...
- 分治 - 计算几何 - BZOJ2458,[BeiJing2011]最小三角形
http://www.lydsy.com/JudgeOnline/problem.php?id=2458 [BeiJing2011]最小三角形 描述 Frisk现在遇到了一个有趣的问题. 平面上有N个 ...
- bzoj2458: [BeiJing2011]最小三角形(分治+几何)
题目链接:bzoj2458: [BeiJing2011]最小三角形 学习推荐博客:分治法编程问题之最接近点对问题的算法分析 题解:先将所有点按x值排列,然后每次将当前区间[l,r]分成左右两半递归求解 ...
- BZOJ2458:[BJOI2011]最小三角形——题解
http://www.lydsy.com/JudgeOnline/problem.php?id=2458 Description Xaviera现在遇到了一个有趣的问题. 平面上有N个点,Xavier ...
- BZOJ2458 Beijing2011最小三角形(分治)
类似于平面最近点对,考虑分治,即分别计算分割线两侧的最小三角形再考虑跨过线的三角形. 复杂度证明也是类似的,对于某一个点,在另一侧可能与其构成最小三角形的点在一个d*d/2的矩形内(两边之和大于第三边 ...
- [BZOJ2458][BeiJing2011]最小三角形(分治)
求平面上n个点组成的周长最小的三角形. 回忆平面最近点对的做法,找到横坐标的中点mid分治到两边,合并时考虑离mid横坐标不超过当前最小值d的所有点,按y排序后暴力更新答案. 这个题也一样,先分治到两 ...
随机推荐
- 牛客网 暑期ACM多校训练营(第二场)I.car-规律思维题
I.car 车只能从一边走到另一边,而且车和车不能相撞,车也不能走到坑里.所以直接找规律,如果没有坑,最多能放多少辆车.就会发现,关于对角线对称的两边只能放一辆车,如果是奇数个的时候,中间的行和中间的 ...
- 网络爬虫框架Webmagic
1 谈谈网络爬虫 1.1 什么是网络爬虫 在大数据时代,信息的采集是一项重要的工作,而互联网中的数据是海量的,如果单纯靠人力进行信息采集,不仅低效繁琐,搜集的成本也会提高.如何自动高效地获取互联网中我 ...
- iOS UI、Xcode、调试、代码等常见问题总汇(持续更新中)
以前比较懒,遇到问题解决了就完事了,有些问题再次遇到时忘记了当初是怎么解决的,又要查各种资料来解决.好记忆不如烂笔头,不管简单还是复杂都记一下吧,所以决定写一篇常见问题总结,方便以后查阅.现在有点忙, ...
- FreeRTOS+FreeModbus+神舟IV号
下面的这个例子是FreeModbus和FreeRTOS在神舟IV号上的应用,仅当做学习用途. 这个demo完成的功能也比较简单,创建了两个任务,一个任务用于控制板子上的LED1,使它每1秒钟闪烁一次. ...
- linux selenium运行chrome
chrome版本要和chromedriver版本匹配才能正常运行.
- nginx的301与302如何配置
转自:http://blog.sina.com.cn/s/blog_5d73ba76010145rr.html 首先看一个完整代码示例,关于nginx 301 302跳转的. 301跳转设置: ser ...
- 通过CVE-2017-17215学习路由器漏洞分析,从入坑到放弃
1.基本信息: 2017/11/27,Check Point 软件技术部门报告了一个华为 HG532 产品的远程命令执行漏洞(CVE-2017-17215),Mirai的升级版变种中已经使用该漏洞.看 ...
- 15个最受欢迎的Python开源框架(转载)
一.Django: Python Web应用开发框架 Django是一个开放源代码的Web应用框架,由Python写成.采用了MVC的软件设计模式,即模型M,视图V和控制器C.它最初是被开发来用于管理 ...
- Java之JDBC学习
(一),MySql数据库 1,MySql数据库的数据类型定义 2,完整性约束: 3,索引: 作用:唯一作用就是加快对表查询速度,索引通过快速路径方法访问来快速定位数据,从而减少磁盘的II/O; 缺点: ...
- Python - 连续替换(replace)的正則表達式(re)
字符串连续替换, 能够连续使用replace, 也能够使用正則表達式. 正則表達式, 通过字典的样式, key为待替换, value为替换成, 进行一次替换就可以. 代码 # -*- coding: ...