BZOJ 1043 [HAOI2008]下落的圆盘 ——计算几何
倒着考虑,加入一个圆,判断和前面有没有完全覆盖的情况。
如果没有,和圆盘一一取交集,然后计算它们的并集,然后计算即可。
#include <map>
#include <cmath>
#include <queue>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define F(i,j,k) for (int i=j;i<=k;++i)
#define D(i,j,k) for (int i=j;i>=k;--i)
#define eps 1e-8
#define sqr(x) x*x
#define ll long long
#define mp make_pair const double pi=acos(-1.0); struct Vector{
double x,y;
void print()
{
printf("Vector -> (%.3f,%.3f)\n",x,y);
}
}; struct Point{
double x,y;
void print()
{
printf("Point . (%.3f,%.3f)\n",x,y);
}
}; Vector operator + (Vector a,Vector b)
{Vector ret;ret.x=a.x+b.x;ret.y=a.y+b.y;return ret;} Vector operator - (Vector a,Vector b)
{Vector ret;ret.x=a.x-b.x;ret.y=a.y-b.y;return ret;} Point operator + (Point a,Vector b)
{Point ret;ret.x=a.x+b.x;ret.y=a.y+b.y;return ret;} Vector operator - (Point a,Point b)
{Vector ret;ret.x=a.x-b.x;ret.y=a.y-b.y;return ret;} Vector operator * (Vector a,double b)
{Vector ret;ret.x=a.x*b;ret.y=a.y*b;return ret;} double operator * (Vector a,Vector b)
{return a.x*b.y-a.y*b.x;} Vector Turn(Vector a,double b)
{
Vector ret;
ret.x=a.x*cos(b)-a.y*sin(b);
ret.y=a.x*sin(b)+a.y*cos(b);
return ret;
} void debug()
{
Vector v1,v2,v3; Point p1,p2,p3;
v1.x=3;v1.y=0;v2.x=0;v2.y=2;
printf("Test one\n");
v1.print(); v2.print(); printf("The Area is %.3f\n\n",v1*v2);
printf("Tet Two\n");
v1.print(); v1=Turn(v1,pi/4);
printf("Turn in π/4\n");
v1.print(); printf("\n");
v1=Turn(v1,-pi/4);
printf("Turn in -π/4\n");
v1.print(); printf("\n");
} int n; struct Circle{
double x,y,r;
void print()
{
printf("Circle (%.3f,%.3f) r = %.3f\n",x,y,r);
}
}a[1005]; double dst(double x1,double y1,double x2,double y2)
{return sqrt((x2-x1)*(x2-x1)+(y2-y1)*(y2-y1));} int dcmp(double x)
{
if (x>eps) return 1;
if (fabs(x)<eps) return 0;
return -1;
} bool in(Circle a,Circle b)
{
if (dcmp(dst(a.x,a.y,b.x,b.y)-(b.r-a.r))<=0) return true;
return false;
} struct Cut_Cir{
double l,r;
void print()
{
printf("Ang (%.3f,%.3f)\n",l,r);
}
}sta[1000005];
int top=0; double nege(double x)
{
if (dcmp(x)==-1) x+=2*pi;
while (dcmp(x-2*pi)>=0) x-=2*pi;
return x;
} void Cut(int pos)
{
top=0; double dist; Vector v0,v1,v2;
F(i,pos+1,n)
if ((dist=dst(a[pos].x,a[pos].y,a[i].x,a[i].y))<a[pos].r+a[i].r)
{
double ang; ang=(sqr(a[pos].r)+sqr(dist)-sqr(a[i].r))/(2*a[pos].r*dist);
if (ang>1.0) continue;
ang=acos(ang);
v0.x=a[i].x-a[pos].x;v0.y=a[i].y-a[pos].y;
v0=v0*(1.0/sqrt(sqr(v0.x)+sqr(v0.y))*a[pos].r);
v1=Turn(v0,ang);
v2=Turn(v0,-ang);
double t1=atan2(v1.y,v1.x),t2=atan2(v2.y,v2.x);
t1+=pi;t2+=pi;
while (t1>2*pi) t1-=2*pi;
while (t2>2*pi) t2-=2*pi;
if (t1<t2)
{
++top;
sta[top].l=0;
sta[top].r=t1;
++top;
sta[top].l=t2;
sta[top].r=2*pi;
}
else
{
++top;
sta[top].l=t2;
sta[top].r=t1;
}
}
} double C(Circle a)
{return a.r*2*pi;} bool cmp2(Cut_Cir a,Cut_Cir b)
{return a.l<b.l;} double cal(double r)
{
sort(sta+1,sta+top+1,cmp2);
double now=0,ret=0;
F(i,1,top)
{
now=max(now,sta[i].l);
ret+=max(0.0,sta[i].r-now);
now=max(now,sta[i].r);
}
return ret*r;
} double ans=0; int main()
{
scanf("%d",&n);
F(i,1,n) scanf("%lf%lf%lf",&a[i].r,&a[i].x,&a[i].y);
F(i,1,n)
{
int flag=1;
F(j,i+1,n) if (in(a[i],a[j]))
{
flag=0;
}
if (!flag) continue;
Cut(i);
ans+=C(a[i]);
ans-=cal(a[i].r);
}
printf("%.3f\n",ans);
}
BZOJ 1043 [HAOI2008]下落的圆盘 ——计算几何的更多相关文章
- BZOJ 1043 HAOI2008 下落的圆盘 计算几何
题目大意:n个圆盘依次下落.求终于能看到的轮廓线面积 円盘反对! 让我们一起团结起来! 赶走円盘! 咳咳.非常神的一道题 今天去看了题解和白书才搞出来-- 首先我们倒着做 对于每一个圆盘处理出在它之后 ...
- bzoj 1043 [HAOI2008]下落的圆盘——圆的周长
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1043 算每个圆被它后面的圆盖住了多少圆弧即可.注意判断这个圆完全被后面盖住的情况. #inc ...
- bzoj1043[HAOI2008]下落的圆盘 计算几何
1043: [HAOI2008]下落的圆盘 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1598 Solved: 676[Submit][Stat ...
- 【BZOJ】1043: [HAOI2008]下落的圆盘(计算几何基础+贪心)
http://www.lydsy.com/JudgeOnline/problem.php?id=1043 唯一让我不会的就是怎么求圆的周长并QAAQ... 然后发现好神!我们可以将圆弧变成$[0, 2 ...
- 1043: [HAOI2008]下落的圆盘 - BZOJ
Description有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求.Input n ri xi y1 ... rn xn y ...
- 【bzoj1043】[HAOI2008]下落的圆盘 计算几何
题目描述 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. 输入 第一行为1个整数n,N<=1000接下来n行每行3个实 ...
- 1043: [HAOI2008]下落的圆盘
Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1725 Solved: 743[Submit][Status][Discuss] Descripti ...
- JZYZOJ1502 [haoi2008]下落的圆盘 计算几何 贪心
http://172.20.6.3/Problem_Show.asp?id=1502这种题用了快一天才写出来也是真的辣鸡.主要思路就是计算一下被挡住的弧度然后对弧度进行贪心.最开始比较困扰的是求弧度值 ...
- 【BZOJ1043】[HAOI2008]下落的圆盘 几何
[BZOJ1043][HAOI2008]下落的圆盘 Description 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周长.看下面这副图, 所有的红色线条的总长度即为所求. ...
随机推荐
- HDU 4276 The Ghost Blows Light (树形DP,变形)
题意:给定一棵n个节点的树,起点是1,终点是n,每经过一条边需要消耗Ti天,每个点上有一定量的珠宝,要求必须在t天内到达终点才可以将珠宝带出去,问至多能带多少珠宝? 思路: 注意Ti可以为0,而且有可 ...
- LeetCode || 递归 / 回溯
呜呜呜 递归好不想写qwq 求“所有情况”这种就递归 17. Letter Combinations of a Phone Number 题意:在九宫格上按数字,输出所有可能的字母组合 Input: ...
- 个人总结NDIS中NDIS_PACKET,NDIS_BUFFER的关系
// // NDIS_PACKET结构的定义 // typedef struct _NDIS_PACKET { NDIS_PACKET_PRIVATE Private; //这个其实是一个链表结构,P ...
- luogu P2574 XOR的艺术 (线段树)
luogu P2574 XOR的艺术 (线段树) 算是比较简单的线段树. 当区间修改时.\(1 xor 1 = 0,0 xor 1 = 1\)所以就是区间元素个数减去以前的\(1\)的个数就是现在\( ...
- syslog命令
更多请关注 Linux命令大全 syslog 介绍 syslog是Linux系统默认的日志守护进程.默认的syslog配置文件是/etc/syslog.conf文件.程序,守护进程和内核提供了访问系统 ...
- 利用js实现图片展开与收缩
1.元素居中放大: 1>除了要改变元素的宽高以外,还要改变元素的定位(left,top),如果图片放大一倍,那么位移放大宽高的一半. 2>元素必须是定位的.所以,在css中设置为浮动布局, ...
- Python基本运算符和流程控制
常量 常量即不可改变的量,在Python中不存在常量,我们只能逻辑上规定一个常量并不去修改它,通常用全大写字母表示. 基本运算符之二 算术运算 运算符 说明 ** 幂运算 *, /, //, % 乘. ...
- 四、10分钟ToPandas_0.24.2
# Author:Zhang Yuan整理,版本Pandas0.24.2 # 0. 习惯上,我们会按下面格式引入所需要的包: import pandas as pd import numpy as n ...
- Python编程快速上手--实践项目11.11.1
from selenium import webdriver from selenium.webdriver.common.keys import Keys import time def messa ...
- Python9-集合-day7
集合是无序的,不重复的数据集合,它里面的元素是可哈希的(不可变类型),但是集合本身是不可哈希(所以集合做不了字典的键)的. 以下是集合最重要的两点: 去重,把一个列表变成集合,就自动去重了. 关系测试 ...