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

2
1 0 0
1 1 0

Sample Output

10.472

————————————————————————————————

代码借(抄)鉴(袭)于: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]下落的圆盘——题解(配图片)的更多相关文章

  1. bzoj1043[HAOI2008]下落的圆盘 计算几何

    1043: [HAOI2008]下落的圆盘 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1598  Solved: 676[Submit][Stat ...

  2. 【计算几何】bzoj1043 [HAOI2008]下落的圆盘

    n^2枚举圆盘,用两圆圆心的向量的极角+余弦定理求某个圆覆盖了该圆的哪一段区间(用弧度表示),最后求个区间并. 注意--精度--最好再累计区间的时候,把每个区间的长度减去EPS,防止最后覆盖的总区间超 ...

  3. bzoj1043 [HAOI2008]下落的圆盘

    Description 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. Input 第一行为1个整数n,N<=1000 ...

  4. BZOJ-1043 [HAOI2008]下落的圆盘

    几何题... 先把所有圆储存起来,然后对于每个圆我们求得之后放下的圆挡住了的部分,求个并集,并把没被挡到的周长加进答案. #include <cstdlib> #include <c ...

  5. 【BZOJ1043】[HAOI2008]下落的圆盘 几何

    [BZOJ1043][HAOI2008]下落的圆盘 Description 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求.  ...

  6. 【bzoj1043】下落的圆盘

    [bzoj1043]下落的圆盘 题意 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. \(1\leq n\leq 1000\ ...

  7. 【BZOJ1043】下落的圆盘 [计算几何]

    下落的圆盘 Time Limit: 10 Sec  Memory Limit: 162 MB[Submit][Status][Discuss] Description 有n个圆盘从天而降,后面落下的可 ...

  8. luogu P2510 [HAOI2008]下落的圆盘

    LINK:下落的圆盘 计算几何.n个圆在平面上编号大的圆将编号小的圆覆盖求最后所有没有被覆盖的圆的边缘的总长度. 在做这道题之前有几个前置知识. 极坐标系:在平面内 由极点 极轴 和 极径组成的坐标系 ...

  9. 【bzoj1043】[HAOI2008]下落的圆盘 计算几何

    题目描述 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. 输入 第一行为1个整数n,N<=1000接下来n行每行3个实 ...

随机推荐

  1. 11gR2RAC更换CRS磁盘组文档

    磁盘(pv)准备     在生产环境中,提前从存储上划分一些磁盘挂载到RAC系统的两个节点上(node1,node2).     新增加磁盘组为(hdisk14--hdisk24) 1.1磁盘使用规划 ...

  2. JAVA FILE.renameTo跨文件系统移动文件失败

    遇到了FILE.renameTo跨文件系统移动文件失败的问题,应使用FILES.move()接口或在同一文件系统移动文件. FILE.renameTo接口说明: public boolean rena ...

  3. 使用union 外加count

    explain extended and name='aaa')) t; +----+--------------+------------+-------+---------------+----- ...

  4. 第十五届北京师范大学程序设计竞赛现场决赛题解&源码(A.思维,C,模拟,水,坑,E,几何,思维,K,字符串处理)

    #include <bits/stdc++.h> using namespace std; int main() { int T,n,a,b; while(cin>>T) { ...

  5. zipaligin的使用介绍

    近来一直在做APK反编译和重编译的工作,针对一些apk需要放入一些相应的文件,(当然这里不涉及非法盈利,都是有合约的),在对一些包打包以后,发现可以通过一个叫做zipalign的工具进行优化,对于这个 ...

  6. 「日常训练」Equation(HDU-5937)

    题意与分析 时隔一个月之后来补题.说写掉的肯定会写掉. 题意是这样的:给1~9这些数字,每个数字有\(X_i\)个,问总共能凑成多少个不同的等式\(A+B=C\)(\(A,B,C\)均为1位,\(1+ ...

  7. 180626-Spring之借助Redis设计一个简单访问计数器

    文章链接:https://liuyueyi.github.io/hexblog/2018/06/26/180626-Spring之借助Redis设计一个简单访问计数器/ Spring之借助Redis设 ...

  8. HDU - 6438(贪心+思维)

    链接:HDU - 6438 题意:给出 n ,表示 n 天.给出 n 个数,a[i] 表示第 i 天,物品的价格是多少.每天可以选择买一个物品,或者卖一个已有物品,也可以什么都不做,问最后最大能赚多少 ...

  9. MFC常用数据类型

    下面这些是和Win32程序共同使用的数据类型BOOL:布尔值,取值为TRUE or FALSEBSTR:32-bit 字符指针BYTE:8-bit整数,未带正负号COLORREF:32-bit数值,代 ...

  10. Beta版本软件使用说明

    北京航空航天大学计算机学院 远航1617 小组 产品版本: Beta版本 产品名称:Crawling   is going on 文档作者:杨帆 文档日期:2013/12/24 1.   引言 1.1 ...