有 n 个圆盘从天而降,后面落下的可以盖住前面的。最后按掉下的顺序,在平面上依次测得每个圆盘的圆心和半径,问下落完成后从上往下看,整个图形的周长是多少,即你可以看到的圆盘的轮廓的圆盘的轮廓总长.例如下图的黑色线条的总长度即为所求。

【输入格式】

第一行为1个整数n

接下来n行每行3个实数,ri,xi,yi,表示下落时第i个圆盘的半径和圆心坐标.

【输出格式】

仅一个实数,表示所求的总周长,答案保留3位小数.

【样例输入】

2
1 0 0
1 1 0

【样例输出】

10.472

【提示】

30%的数据,n<=10

100%的数据,n<=1000

数学问题 计算几何

用余弦定理和三角函数可以计算出两圆相交部分的弧长。

对于每个圆,计算它和在它之后落下的所有圆的角。记录相交部分的弧对应的圆心角范围。将“圆心角”区间离散到0~2pi的数轴上,做线段覆盖。

知道了未被覆盖的角总共有多大,就能算出该圆未被覆盖的弧有多长。

注意如果求出的覆盖部分圆心角范围超出了0~2pi,要变换到0~2pi范围内

 #include<iostream>
#include<algorithm>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<vector>
using namespace std;
const double pi=acos(-1.0);
const double eps=1e-;
const int mxn=;
struct point{
double x,y;
point operator + (point b){return (point){x+b.x,y+b.y};}
point operator - (point b){return (point){x-b.x,y-b.y};}
double operator * (point b){return x*b.x+y*b.y;}
};
inline double Cross(point a,point b){
return a.x*b.y-a.y*b.x;
}
inline double dist(point a,point b){
return sqrt((a.x-b.x)*(a.x-b.x)+(a.y-b.y)*(a.y-b.y));
}
inline double Len(point a){return sqrt(a*a);}
struct cir{
double x,y;
double r;
point operator + (cir b){return (point){x+b.x,y+b.y};}
point operator - (cir b){return (point){x-b.x,y-b.y};}
}c[mxn];
inline bool cover(cir a,cir b){
return (a.r>=b.r+Len(a-b));
}
struct line{
double l,r;
bool operator < (line b)const{
return (l<b.l)|| (l==b.l && r<b.r);
}
};
line CX(cir a,cir b){
double dis=Len(b-a);
double angle=acos((a.r*a.r+dis*dis-b.r*b.r)/(*a.r*dis));
double deg=atan2(a.x-b.x,a.y-b.y);//统一旋转pi角度,保证在-2pi~2pi范围内
return (line){deg-angle,deg+angle};
/* double t=(a.r*a.r+dis*dis-b.r*b.r)/(2*dis);
double st=atan2(a.x-b.x,a.y-b.y);
double l=acos(t/a.r);
return (line){st-l,st+l};*/
}
vector<line>ve;
int n;
double ans=;
void calc(int x){
for(int i=x+;i<=n;i++){if(cover(c[i],c[x]))return;}//被完全覆盖
ve.clear();
for(int i=x+;i<=n;i++){
if(cover(c[x],c[i]))continue;//完全覆盖
line tmp;
if(c[x].r+c[i].r>Len(c[i]-c[x])) tmp=CX(c[x],c[i]);
else continue;
if(tmp.l<) tmp.l+=*pi;
if(tmp.r<) tmp.r+=*pi;
if(tmp.l>tmp.r){//拆分
ve.push_back((line){,tmp.r});
ve.push_back((line){tmp.l,*pi});
}
else ve.push_back(tmp);
}
sort(ve.begin(),ve.end());
// printf("mid\n");
double now=,ran=;
for(int i=;i<ve.size();i++){//线段覆盖
line tmp=ve[i];
// printf("i:%d %.3f %.3f\n",i,tmp.l,tmp.r);
if(tmp.l>now){ran+=tmp.l-now;now=tmp.r;}
else now=max(now,tmp.r);
}
ran+=*pi-now;
ans+=c[x].r*ran;//累加半径
return;
}
int main(){
freopen("disc.in","r",stdin);
freopen("disc.out","w",stdout);
int i,j;
scanf("%d",&n);
for(i=;i<=n;i++){
scanf("%lf%lf%lf",&c[i].r,&c[i].x,&c[i].y);
}
for(i=n;i>=;i--){
calc(i);
// printf("ans:%.3f\n",ans);
}
printf("%.3f\n",ans);
return ;
}

Bzoj1313 [HAOI2008]下落的圆盘的更多相关文章

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

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

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

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

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

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

  4. BZOJ 1043 HAOI2008 下落的圆盘 计算几何

    题目大意:n个圆盘依次下落.求终于能看到的轮廓线面积 円盘反对! 让我们一起团结起来! 赶走円盘! 咳咳.非常神的一道题 今天去看了题解和白书才搞出来-- 首先我们倒着做 对于每一个圆盘处理出在它之后 ...

  5. [HAOI2008]下落的圆盘

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

  6. bzoj1043 [HAOI2008]下落的圆盘

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

  7. BZOJ1043:[HAOI2008]下落的圆盘——题解(配图片)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1043 Description 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周 ...

  8. 1043: [HAOI2008]下落的圆盘

    Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1725  Solved: 743[Submit][Status][Discuss] Descripti ...

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

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

随机推荐

  1. java冒泡算法

    public static void maopao() { int temp; int[] values = {2, 34, 0, 245}; for (int i = 0; i < value ...

  2. Python的文件输入输出,如何追加内容,读取内容,添加内容

    python主要的文件打开的几种访问模式 模式可以为读模式('r').写模式('w')或追加模式('a'),当然还有rb.wb.ab.r+.w+.a+.rb+.wb+.ab+,然而实际从代码上我也没看 ...

  3. list 集合addAll 和 add 方法小坑

    1.问题 我们经常会遍历 list集合,在遍历的过程中,如果在遍历的过程中添加了 add()  或者 addAll() 方法修改了遍历的list列表,那么会报错. 代码演示: List<Inte ...

  4. Qscintilla2编译使用

    Qscintilla2的下载地址: https://github.com/josephwilk/qscintilla https://riverbankcomputing.com/software/q ...

  5. Source Tree基础教程2

    1.分支 项目——分支——推送 新分支要重新拉取项目后才可以看见 项目——拉取 2合并分支代码 将其他分支代码合并到当前分支——提交

  6. java05笔记

  7. Java面试题(下)

    这部分主要是开源Java EE框架方面的内容,包括hibernate.MyBatis.spring.Spring MVC等,由于Struts 2已经是明日黄花,在这里就不讨论Struts 2的面试题, ...

  8. JavaScript中常用转义字符

    \b   退格 \f   换页 \r   回车 \n   换行 \"   双引号 \'  单引号 \t  Tab字符 \\  反斜杠 \xnn  十六进制代码nn表示的字符 \unnnn 十 ...

  9. DataTable定义

    DataTable是一个临时保存数据的网格虚拟表(表示内存中数据的一个表.).DataTable是ADO dot net 库中的核心对象.它可以被应用在 VB 和 ASP 上.它无须代码就可以简单的绑 ...

  10. Win10间歇性卡顿

    Win10间歇性卡顿 1.关闭不必要的服务:Windows Update.Windows Search.SuperFetch.Background Intelligent Transfer Servi ...