【BZOJ】1043: [HAOI2008]下落的圆盘(计算几何基础+贪心)
http://www.lydsy.com/JudgeOnline/problem.php?id=1043
唯一让我不会的就是怎么求圆的周长并QAAQ...
然后发现好神!我们可以将圆弧变成$[0, 2 \pi ]$的直线!
然后一定要注意!起点是$(1, 0)$(单位圆)
首先学了余弦定理...
在三角形ABC中
$$cos A=\frac{|AB|^2+|AC|^2-|BC|^2}{2|AB| |AC|}$$
证明很简单...
$$
\begin{align}
|{BC}|^2 & = \vec{BC} \cdot \vec{BC} \\
& = (\vec{AC}-\vec{AB}) \cdot (\vec{AC}-\vec{AB}) \\
& = | \vec{AC} |^2 + | \vec{AB} |^2 - 2|AC| |AB|cos A \\
\end{align}
$$
移项一下就是:
$$cos A=\frac{|AB|^2+|AC|^2-|BC|^2}{2|AB| |AC|}$$
一开始看了题解后想了一下就写了,求圆的交点..交点的极角...然后离散到的是$[0, \pi ]$的直线...然后喜闻乐见的悲剧了...
一定要知道!极角虽然是算出来了!但是同一个位置可能有两种角啊....为嘛我当时脑残...${\alpha + 2k \pi}$啊....
所以要区分一下..当极角<0时,先加上$2\pi$,然后如果原来极角小的比原来极角大的角要大了,说明它要分成两段来计算,即$[0, r]$, $[l, 2\pi ]$
然后将线段排序贪心取即可..(其实可以直接差分啊QAQ我是sb...
#include <cstdio>
#include <cstring>
#include <cmath>
#include <string>
#include <iostream>
#include <algorithm>
#include <queue>
#include <set>
#include <map>
using namespace std;
typedef long long ll;
#define rep(i, n) for(int i=0; i<(n); ++i)
#define for1(i,a,n) for(int i=(a);i<=(n);++i)
#define for2(i,a,n) for(int i=(a);i<(n);++i)
#define for3(i,a,n) for(int i=(a);i>=(n);--i)
#define for4(i,a,n) for(int i=(a);i>(n);--i)
#define CC(i,a) memset(i,a,sizeof(i))
#define read(a) a=getint()
#define print(a) printf("%d", a)
#define dbg(x) cout << (#x) << " = " << (x) << endl
#define error(x) (!(x)?puts("error"):0)
#define rdm(x, i) for(int i=ihead[x]; i; i=e[i].next)
inline const int getint() { int r=0, k=1; char c=getchar(); for(; c<'0'||c>'9'; c=getchar()) if(c=='-') k=-1; for(; c>='0'&&c<='9'; c=getchar()) r=r*10+c-'0'; return k*r; } const double PI=acos(-1), eps=1e-6;
int dcmp(double x) { return abs(x)<eps?0:(x<0?-1:1); }
struct ipt { double x, y; ipt(double _x=0, double _y=0) : x(_x), y(_y) {} };
struct icir { ipt x; double r; };
typedef ipt ivt;
ivt operator- (ipt &a, ipt &b) { return ivt(a.x-b.x, a.y-b.y); }
double iangle(ivt v) { return atan2(v.y, v.x); }
double sqr(double x) { return x*x; }
double dis(ipt &a, ipt &b) { return sqrt(sqr(a.x-b.x)+sqr(a.y-b.y)); } bool getCC(icir &a, icir &b, double &ang1, double &ang2) {
static double d, ang;
d=dis(a.x, b.x);
if(dcmp(d)==0) return false;
if(dcmp(a.r+b.r-d)<=0) return false;
if(dcmp(a.r-b.r-d)>=0) return false;
ang=iangle(b.x-a.x);
double da=acos((sqr(d)+sqr(a.r)-sqr(b.r))/a.r/d/2);
ang1=ang-da; //dbg(da);
ang2=ang+da;
return true;
} const int N=1005;
int n;
icir a[N];
struct dat { double l, r; }line[N*2];
bool cmp(const dat &a, const dat &b) { return a.l==b.l?a.r>b.r:a.l<b.l; } double work(int now) {
int ln=0;
static double ang1, ang2;
for1(i, now+1, n) { double d=dis(a[now].x, a[i].x); if(dcmp(a[i].r-a[now].r-d)>=0) return 0; } //puts("here");
for1(i, now+1, n) {
if(getCC(a[now], a[i], ang1, ang2)==false) continue;
if(ang1<0) ang1+=PI*2;
if(ang2<0) ang2+=PI*2;
if(dcmp(ang1-ang2)>0) {
++ln; line[ln].l=0; line[ln].r=ang2;
++ln; line[ln].l=ang1; line[ln].r=PI*2;
}
else { ++ln; line[ln].l=ang1; line[ln].r=ang2; }
}
sort(line+1, line+1+ln, cmp);
double w=0, mx=0;
for1(i, 1, ln) {
if(dcmp(mx-line[i].r)>=0) continue;
mx=max(mx, line[i].l);
w+=line[i].r-mx;
mx=max(mx, line[i].r);
}
return a[now].r*(PI*2-w);
} int main() {
read(n);
for1(i, 1, n) scanf("%lf%lf%lf", &a[i].r, &a[i].x.x, &a[i].x.y);
double ans=0;
for1(i, 1, n) ans+=work(i);
printf("%.3f\n", ans);
return 0;
}
Description
有n个圆盘从天而降,后面落下的可以盖住前面的。求最后形成的封闭区域的周长。看下面这副图, 所有的红色线条的总长度即为所求. 
Input
n ri xi y1 ... rn xn yn
Output
最后的周长,保留三位小数
Sample Input
1 0 0
1 1 0
Sample Output
HINT
数据规模
n<=1000
Source
【BZOJ】1043: [HAOI2008]下落的圆盘(计算几何基础+贪心)的更多相关文章
- BZOJ 1043 HAOI2008 下落的圆盘 计算几何
题目大意:n个圆盘依次下落.求终于能看到的轮廓线面积 円盘反对! 让我们一起团结起来! 赶走円盘! 咳咳.非常神的一道题 今天去看了题解和白书才搞出来-- 首先我们倒着做 对于每一个圆盘处理出在它之后 ...
- BZOJ 1043 [HAOI2008]下落的圆盘 ——计算几何
倒着考虑,加入一个圆,判断和前面有没有完全覆盖的情况. 如果没有,和圆盘一一取交集,然后计算它们的并集,然后计算即可. #include <map> #include <cmath& ...
- bzoj 1043 [HAOI2008]下落的圆盘——圆的周长
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1043 算每个圆被它后面的圆盖住了多少圆弧即可.注意判断这个圆完全被后面盖住的情况. #inc ...
- 1043: [HAOI2008]下落的圆盘 - BZOJ
Description有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求.Input n ri xi y1 ... rn xn y ...
- 1043: [HAOI2008]下落的圆盘
Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1725 Solved: 743[Submit][Status][Discuss] Descripti ...
- bzoj1043[HAOI2008]下落的圆盘 计算几何
1043: [HAOI2008]下落的圆盘 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1598 Solved: 676[Submit][Stat ...
- 【BZOJ1043】[HAOI2008]下落的圆盘 几何
[BZOJ1043][HAOI2008]下落的圆盘 Description 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. ...
- 题解:[HAOI2008]下落的圆盘
时空限制:1000ms / 128MB 原题链接: 洛谷 bzoj Description 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红 色线条的 ...
- luogu P2510 [HAOI2008]下落的圆盘
LINK:下落的圆盘 计算几何.n个圆在平面上编号大的圆将编号小的圆覆盖求最后所有没有被覆盖的圆的边缘的总长度. 在做这道题之前有几个前置知识. 极坐标系:在平面内 由极点 极轴 和 极径组成的坐标系 ...
随机推荐
- 我常用的delphi 第三方控件
转载:http://www.cnblogs.com/xalion/archive/2012/01/09/2317246.html 有网友问我常用的控件及功能.我先大概整理一下,以后会在文章里面碰到时再 ...
- (转)女生应该找一个玩ACM的男生
1.强烈的事业心 将来,他也一定会有自己热爱的事业.而且,男人最性感的时刻之一,就是他专心致志做事的时候.所以,找一个机会在他全神贯注玩ACM的时候,从侧面好好观察他,你就会发现我说的话没错. 2.永 ...
- C++中new的解说
new int;//开辟一个存放整数的存储空间,返回一个指向该存储空间的地址(即指针) new int(100);//开辟一个存放整数的空间,并指定该整数的初值为100,返回一个指向该存储空间的地址 ...
- 【转】如何调试bash脚本
本文转自:http://coolshell.cn/articles/1379.html Bash 是Linux操作系统的默认Shell脚本.Shell是用来处理操作系统和用户交互的一个程序.Shell ...
- linux自定义脚本添加到rc.local脚本无法正常运行的问题
为了能科学地上网,你懂的.其中需要将服务端做成开机启动.然而脚本在secure crt下能正常运行,添加到/etc/rc.local下却无法正常启动服务.用ps查找了下,脚本是运行了,但服务没起来.于 ...
- kettle job通过javascript进行循环控制
任何一种编程语言都少不了循环,kettle中的job也一样.那么kettle中的job是怎么通过JavaScript来达到类似于编程语言中的for循环呢? var max = parent_job.g ...
- 对Java中字符串的进一步理解
字符串在程序开发中无处不在,也是用户交互所涉及到最频繁的数据类型,那么字符串不仅仅就是我们简单的理解的String str = "abc";一起来更加深入的看一下 在Java中,字 ...
- android 中如何分析内存泄漏
转载:http://blog.csdn.net/fulinwsuafcie/article/details/8363218 前提条件: 1,电脑安装了java 运行环境 2,手机端开启了 USB 调试 ...
- js打印html中的内容
js打印方法 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www. ...
- 微信支付v3发布到iis时的证书问题(转)
本文纯粹转载(原地址:微信支付v3发布到iis时的证书问题 ) 一开始报“出现了内部错误” 解决方法是 方法一 var cer = new X509Certificate(certpath, pass ...