BZOJ1043:[HAOI2008]下落的圆盘——题解(配图片)
http://www.lydsy.com/JudgeOnline/problem.php?id=1043
Description
有n个圆盘从天而降,后面落下的可以盖住前面的。求最后形成的封闭区域的周长。看下面这副图, 所有的红
色线条的总长度即为所求. 
Input
第一行为1个整数n,N<=1000
接下来n行每行3个实数,ri,xi,yi,表示下落时第i个圆盘的半径和圆心坐标.
Output
最后的周长,保留三位小数
Sample Input
1 0 0
1 1 0
Sample Output
————————————————————————————————
代码借(抄)鉴(袭)于:http://blog.csdn.net/Vmurder/article/details/46564199
首先我们通过枚举判断两圆的关系:后全覆盖先,先全覆盖后,后和先相交,后和先相离。
显然2和4是没有影响的,而1相当于将先圆干掉了(因为它不再对答案有贡献了)
所以重点在3情况上。
我们弧长公式有:L=角(弧度制)*半径(R)。
所以我们可以用弧度制来表示当前圆被覆盖的部分,即可求出弧长。
基本上高中数学知识即可解决,这里配一张图:

我们这里让a圆覆盖b圆,求b被覆盖的圆心角区间。
我们首先发现图中所有的线段长度都能求出来。
我们设∠EBA为alpha,显然△ADB和△ACB全等,则设∠DBA=∠CBA=beta
那么
alpha=arctan(AE/EB)
beta=arccos((BD*BD+BA*BA-DA*DA)/(2*BD*BA))=arccos((rb*rb+dis*dis-ra*ra)/(2*rb*dis))
//余弦定理
那么∠DBE=alpha-beta,∠EBC=alpha+beta
(这里可以发现我们圆心角的0度被我们定义在了左边,和常识不同请注意)
我们将圆心角控制在[-180度,180度],所以一旦超过了这个区间我们就要对其进行修改。
修改操作详见gai函数。
#include<cstdio>
#include<queue>
#include<cctype>
#include<cstring>
#include<stack>
#include<cmath>
#include<algorithm>
using namespace std;
typedef double dl;
const int N=;
const dl poi=acos(-1.0);
struct circle{
dl r;
dl x;
dl y;
}p[N];
struct line{
dl l;
dl r;
}seg[N][*N];
int n,cnt[N];
bool die[N];
inline dl dis(circle a,circle b){
return sqrt((a.x-b.x)*(a.x-b.x)+(a.y-b.y)*(a.y-b.y));
}
inline int inc(circle a,circle b){
dl d=dis(a,b);
if(a.r+b.r<d)return ;//两圆相离
if(a.r>b.r&&a.r-b.r>d)return -;//a覆盖b
if(b.r>a.r&&b.r-a.r>d)return ;//b覆盖a
return ;//两圆相交
}
inline void getinc(circle a,circle b,dl &i,dl &j){//a覆盖b
double alpha=atan2((b.y-a.y),(b.x-a.x));
dl d=dis(a,b);
double beta=acos((b.r*b.r+d*d-a.r*a.r)/(*b.r*d));
i=alpha-beta;
j=alpha+beta;
return;
}
inline bool gai(line &a){
if(a.r>poi){
a.r-=*poi;
return ;
}
if(a.l<-poi){
a.l+=*poi;
return ;
}
return ;
}
bool cmp(line a,line b){
return a.l<b.l;
}
int main(){
scanf("%d",&n);
for(int i=;i<=n;i++){
scanf("%lf%lf%lf",&p[i].r,&p[i].x,&p[i].y);
for(int j=;j<i;j++){
if(die[j])continue;
int k=inc(p[i],p[j]);
if(!k)continue;
if(k==-){
die[j]=;
continue;
}
cnt[j]++;
getinc(p[i],p[j],seg[j][cnt[j]].l,seg[j][cnt[j]].r);
if(gai(seg[j][cnt[j]])){
cnt[j]++;
seg[j][cnt[j]].l=-poi;
seg[j][cnt[j]].r=seg[j][cnt[j]-].r;
seg[j][cnt[j]-].r=poi;
}
}
}
dl ans=;
for(int i=;i<=n;i++){
if(!die[i]){
dl re=*poi,L,R;
if(cnt[i]){
sort(seg[i]+,seg[i]+cnt[i]+,cmp);
L=seg[i][].l;R=seg[i][].r;
for(int j=;j<=cnt[i];j++){
if(seg[i][j].l>R){
re-=R-L;
L=seg[i][j].l;
R=seg[i][j].r;
}else{
R=max(R,seg[i][j].r);
}
}
re-=R-L;
}
ans+=re*p[i].r;
}
}
printf("%.3lf\n",ans);
return ;
}
BZOJ1043:[HAOI2008]下落的圆盘——题解(配图片)的更多相关文章
- bzoj1043[HAOI2008]下落的圆盘 计算几何
1043: [HAOI2008]下落的圆盘 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1598 Solved: 676[Submit][Stat ...
- 【计算几何】bzoj1043 [HAOI2008]下落的圆盘
n^2枚举圆盘,用两圆圆心的向量的极角+余弦定理求某个圆覆盖了该圆的哪一段区间(用弧度表示),最后求个区间并. 注意--精度--最好再累计区间的时候,把每个区间的长度减去EPS,防止最后覆盖的总区间超 ...
- bzoj1043 [HAOI2008]下落的圆盘
Description 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. Input 第一行为1个整数n,N<=1000 ...
- BZOJ-1043 [HAOI2008]下落的圆盘
几何题... 先把所有圆储存起来,然后对于每个圆我们求得之后放下的圆挡住了的部分,求个并集,并把没被挡到的周长加进答案. #include <cstdlib> #include <c ...
- 【BZOJ1043】[HAOI2008]下落的圆盘 几何
[BZOJ1043][HAOI2008]下落的圆盘 Description 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. ...
- 【bzoj1043】下落的圆盘
[bzoj1043]下落的圆盘 题意 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. \(1\leq n\leq 1000\ ...
- 【BZOJ1043】下落的圆盘 [计算几何]
下落的圆盘 Time Limit: 10 Sec Memory Limit: 162 MB[Submit][Status][Discuss] Description 有n个圆盘从天而降,后面落下的可 ...
- luogu P2510 [HAOI2008]下落的圆盘
LINK:下落的圆盘 计算几何.n个圆在平面上编号大的圆将编号小的圆覆盖求最后所有没有被覆盖的圆的边缘的总长度. 在做这道题之前有几个前置知识. 极坐标系:在平面内 由极点 极轴 和 极径组成的坐标系 ...
- 【bzoj1043】[HAOI2008]下落的圆盘 计算几何
题目描述 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. 输入 第一行为1个整数n,N<=1000接下来n行每行3个实 ...
随机推荐
- Linux用户切换和密码修改
1.普通用户切换到root su - 再输入root密码,密码正确,成功切换,再输入exit则切换回普通用户 2.root切换到其他用户,例user su - user 再输入exit,则切换回roo ...
- darknet 识别获取结果
在examples/darknet.c文件中若使用detect命令可以看到调用了test_detector. ... else if (0 == strcmp(argv[1], "detec ...
- Python常用函数--文档字符串DocStrings
Python 有一个甚是优美的功能称作python文档字符串(Documentation Strings),在称呼它时通常会使用另一个短一些的名字docstrings.DocStrings 是一款你应 ...
- 【rich-text】 富文本组件说明
[rich-text] 富文本组件可以显示HTML代码样式. 1)支持事件:tap.touchstart.touchmove.touchcancel.touchend和longtap 2)信任的HTM ...
- LeetCode 386——字典序排数
1. 题目 2. 解答 2.1 方法一 假设返回 118 以内数的字典顺序,则为 1,10,100,101,102,...,109,11,110,111,112,...,118,12,13,....根 ...
- Python3 标准库:calendar,time
1.calendar import calendar print(calendar.month(2008,8)) #某个月 print(calendar.calendar(2008)) #某年 pri ...
- sqoop-1.4.6安装与使用
一.安装 1.下载sqoop-1.4.6-bin.tar.gz并解压 2.修改conf/sqoop-env.sh,设置如下变量: export HADOOP_COMMON_HOME=/usr/loca ...
- [C++] OOP - Access Control and Class Scope
Access Control And Inheritance Protected Member Like private, protected members are unaccessible to ...
- 《剑指Offer》题三十一~题四十
三十一.栈的压入.弹出序列 题目:输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否为该栈的弹出顺序.假设压入栈的数字均不相等.例如,序列{1, 2, 3, 4 ,5}是某栈的压栈序列 ...
- 计算器软件实现系列(六)windowform窗体+SQL+策略模式
一 整体概述 这个计算器软件的功能和以前的功能基本上一样,只不过是数据的保存形式发生了变化,,以前用的是txt文件保存,现在更正用SQL数据库,现在更改了以前的文件保存形式,是三层架构中数据层的更换, ...