JZYZOJ1502 [haoi2008]下落的圆盘 计算几何 贪心

http://172.20.6.3/Problem_Show.asp?id=1502
这种题用了快一天才写出来也是真的辣鸡。主要思路就是计算一下被挡住的弧度然后对弧度进行贪心。
最开始比较困扰的是求弧度值及其起始位置的部分,弧度值很好求,位置有点恶心,我的起始位置设置的是圆的十二点方向顺时针到起始位置的弧度值,然后我分了四种情况讨论(因为遮挡的方向有两种不同情况,遮挡部分弧度值与180度的关系又是两种情况),应该是有更简单的方法的,但是我只能想出来这种了。。。

代码

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<algorithm>
 4 #include<cmath>
 5 #include<iostream>
 6 using namespace std;
 7 const int maxn=1010;
 8 const double eps=0.0000001;
 9 long long n;
10 double r[maxn]={},x[maxn]={},y[maxn]={};
11 struct nod{
12     double id,v;
13 }e[maxn];
14 int tot;
15 bool mmp(nod aa,nod bb){
16     return aa.id<bb.id;
17 }
18 double getit(int i){
19     tot=0;double cnt=0;
20     for(int j=n;j>i;j--){
21         double w=sqrt((x[i]-x[j])*(x[i]-x[j])+(y[i]-y[j])*(y[i]-y[j]));
22         if(r[i]+r[j]-w<=0||r[i]-r[j]-w>=0)continue;
23         if(r[j]-r[i]-w>=0) return 0;
24
25         double z=(r[i]*r[i]-r[j]*r[j]+w*w)/(2*w);
26         double h=acos(z/r[i])*2;
27
28         double xx=x[i]+(x[j]-x[i])/w*z,yy=y[i]+(y[j]-y[i])/w*z;
29         double w1=sqrt((x[i]-xx)*(x[i]-xx)+(y[i]+z-yy)*(y[i]+z-yy));
30         double h2=asin(w1/2/z)*2;
31         if(xx<x[i]){
32             if(h2<0)h2=-h2;
33             else if(h2>0) h2=2*M_PI-h2; }
34         else if (xx>x[i]){if(h2<0) h2=2*M_PI+h2;}
35         h2=h2-h/2;
36         if(h2<0)h2+=2*M_PI;
37         h=h2+h;
38         if(h>2*M_PI){
39             e[++tot].id=h2;e[tot].v=2*M_PI;
40             e[++tot].id=0;e[tot].v=h-2*M_PI;
41         }else{e[++tot].id=h2;e[tot].v=h;}
42     }
43     sort(e+1,e+1+tot,mmp);
44     double ll=0.0,rr=0.0;
45     for(int j=1;j<=tot;j++){
46         if(e[j].id-rr<0){
47             if(e[j].v-rr>0){
48                 rr=e[j].v;
49             }
50         }
51         else if(e[j].v-rr>0){
52             cnt+=rr-ll;
53             rr=e[j].v;ll=e[j].id;
54         }
55     }
56     cnt+=rr-ll;
57     return r[i]*(M_PI*2-cnt);
58 }
59 int main(){
60     //freopen("wtf.in","r",stdin);
61     scanf("%d",&n);
62     for(int i=1;i<=n;i++){
63         scanf("%lf%lf%lf",&r[i],&x[i],&y[i]);
64     }double ans=0;
65     for(int i=n;i>=1;i--){
66         ans+=getit(i);
67     }printf("%.3f\n",ans);
68     return 0;
69 }

时间: 2024-12-30 10:15:17

JZYZOJ1502 [haoi2008]下落的圆盘 计算几何 贪心的相关文章

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

1043: [HAOI2008]下落的圆盘 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1598  Solved: 676[Submit][Status][Discuss] Description 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求.  Input 第一行为1个整数n,N<=1000接下来n行每行3个实数,ri,xi,yi,表示下落时第i个圆盘的半径和圆心坐标

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

题目描述 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. 输入 第一行为1个整数n,N<=1000接下来n行每行3个实数,ri,xi,yi,表示下落时第i个圆盘的半径和圆心坐标. 输出 最后的周长,保留三位小数 样例输入 2 1 0 0 1 1 0 样例输出 10.472 题解 计算几何 考虑从下到上的每一个圆,它被其它的圆覆盖了多少.即考虑它被覆盖了多少弧度. 考虑两个圆,如果相离则不覆盖,内含判断一下包含关系. 如果

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

题目大意:n个圆盘依次下落.求终于能看到的轮廓线面积 円盘反对! 让我们一起团结起来! 赶走円盘! 咳咳.非常神的一道题 今天去看了题解和白书才搞出来-- 首先我们倒着做 对于每一个圆盘处理出在它之后落下的圆盘和它的覆盖区间 然后求一个区间并就能算出这个圆盘的可见弧长 然后就是相交部分怎么求的问题了 首先两个圆必须相交 然后作圆心1到圆心2的向量 用atan2求出极角 然后利用余弦定理求出两个交点和圆心连线的夹角就可以 注意区间不在[0,2π]的部分要切割成还有一个区间 处理起来事实上不是非常麻

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

[BZOJ1043][HAOI2008]下落的圆盘 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 题解:对于每个圆,我们枚举它后面的

【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{

【BZOJ1043】【HAOI2008】下落的圆盘 计算几何

链接: #include <stdio.h> int main() { puts("转载请注明出处[辗转山河弋流歌 by 空灰冰魂]谢谢"); puts("网址:blog.csdn.net/vmurder/article/details/46564199"); } 题解: 给每个圆求一下: 1. 它是不是被之后的某圆整体覆盖. 2. 它的圆周上有哪些弧段被覆盖了. 然后对于每个圆求一下还剩多少周长即可. 上述的"2."可以用圆的圆心角

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 2 1 0 0 1 1 0 Sample Out

BZOJ 1043: [HAOI2008]下落的圆盘

Description 求几个圆交起来的周长..\(n\leqslant 10^3\) Solution 计算几何. 圆圆求交.. Code /************************************************************** Problem: 1043 User: BeiYu Language: C++ Result: Accepted Time:520 ms Memory:1308 kb *******************************

BZOJ1043 [HAOI2008]下落的圆盘

倒过来做,然后就变成了线段覆盖问题了. 线段覆盖就是贪心即可... 但是好烦好烦= =,需要耐心和几何基础2333 1 /************************************************************** 2 Problem: 1043 3 User: rausen 4 Language: C++ 5 Result: Accepted 6 Time:240 ms 7 Memory:872 kb 8 ****************************