SPOJ CIRU
SPOJ CIRU
题意
给出n个圆,求他们覆盖的面积。
解法
自适应Simpson,但需要将圆离散化一下,以保证我们查询的是一个连续的有圆的区间。 奇怪的是我没有离散化,样例都没有过,却把题给A了
代码如下: ( 注意 :要去掉被覆盖的圆,才不会TLE)
#include <bits/stdc++.h>
using namespace std;
const int maxn=1000+5;
struct cir {
double x,y,r;
cir(double x=0.0,double y=0.0,double r=0.0):x(x),y(y),r(r){}
double dh(double nowx) {
return sqrt( r*r - (nowx - x)*(nowx - x) );
}
bool operator < (const cir&a) const {
return r<a.r;
}
}c[maxn];
struct line {
double y;
bool k;
bool operator < (const line& a) const {
return y < a.y;
}
}temp[maxn];
int n;
double F(double x) {
int num=0;double ans=0.0;
for(int i=1;i<=n;i++) {
if(x>=c[i].x-c[i].r&&x<=c[i].x+c[i].r){
double dy=c[i].dh(x);
temp[++num].y=c[i].y-dy; temp[num].k=1;
temp[++num].y=c[i].y+dy; temp[num].k=0;
}
}
sort(temp+1,temp+1+num);
int f=0;double be;
for(int i=1;i<=num;i++) {
if(temp[i].k) {
if(!(f++)) {be=temp[i].y;}
}
else if(!(--f)) ans+=temp[i].y-be;
}
return ans;
}
double simpson(double a,double b) {
double mid=(a+b)/2;
return ( F(a) + 4*F(mid) + F(b) ) * (b - a) / 6;
}
double asr(double a,double b,double e,double A) {
double mid=(a+b)/2;
double L=simpson(a,mid),R=simpson(mid,b);
if( fabs(L+R-A) <= 15*e) return L+R+(L+R-A)/15;
return asr(a,mid,e/2,L)+asr(mid,b,e/2,R);
}
double asr(double l,double r,double eps) {
return asr(l,r,eps,simpson(l,r));
}
double dis(int a,int b){
return sqrt( (c[a].x-c[b].x)*(c[a].x-c[b].x)+(c[a].y-c[b].y)*(c[a].y-c[b].y) );
}
bool cover[maxn];
int main() {
scanf("%d",&n);
double l=2e33,r=-2e33;
for(int i=1;i<=n;i++) {
scanf("%lf%lf%lf",&c[i].x,&c[i].y,&c[i].r);
l=min(l,c[i].x-c[i].r);
r=max(r,c[i].x+c[i].r);
}
sort(c+1,c+1+n);
int cnt=n;n=0;
for(int i=1;i<cnt;i++)
for(int j=i+1;j<=cnt;j++){
if(dis(i,j)+c[i].r<=c[j].r){
cover[i]=1;
break;
}
}
for(int i=1;i<=cnt;i++)
if(!cover[i])
c[++n]=c[i];
printf("%.3lf",asr(l,r,1e-4));
return 0;
}
以下是加了离散化过后的版本。
#include <bits/stdc++.h>
using namespace std;
const int maxn=1000+5;
struct cir {
double x,y,r;
cir(double x=0.0,double y=0.0,double r=0.0):x(x),y(y),r(r){}
double dh(double nowx) {
return sqrt( r*r - (nowx - x)*(nowx - x) );
}
bool operator < (const cir&a) const {
return r<a.r;
}
}c[maxn];
struct line {
double y;
bool k;
bool operator < (const line& a) const {
return y < a.y;
}
}temp[maxn << 1],k[maxn << 1];
int n;
double F(double x) {
int num=0;double ans=0.0;
for(int i=1;i<=n;i++) {
if(x>=c[i].x-c[i].r&&x<=c[i].x+c[i].r){
double dy=c[i].dh(x);
temp[++num].y=c[i].y-dy; temp[num].k=1;
temp[++num].y=c[i].y+dy; temp[num].k=0;
}
}
sort(temp+1,temp+1+num);
int f=0;double be;
for(int i=1;i<=num;i++) {
if(temp[i].k) {
if(!(f++)) {be=temp[i].y;}
}
else if(!(--f)) ans+=temp[i].y-be;
}
return ans;
}
double simpson(double a,double b) {
double mid=(a+b)/2;
return ( F(a) + 4*F(mid) + F(b) ) * (b - a) / 6;
}
double asr(double a,double b,double e,double A) {
double mid=(a+b)/2;
double L=simpson(a,mid),R=simpson(mid,b);
if( fabs(L+R-A) <= 15*e) return L+R+(L+R-A)/15;
return asr(a,mid,e/2,L)+asr(mid,b,e/2,R);
}
double asr(double l,double r,double eps) {
return asr(l,r,eps,simpson(l,r));
}
double dis(int a,int b){
return sqrt( (c[a].x-c[b].x)*(c[a].x-c[b].x)+(c[a].y-c[b].y)*(c[a].y-c[b].y) );
}
bool cover[maxn];
int main() {
scanf("%d",&n);
for(int i=1;i<=n;i++) {
scanf("%lf%lf%lf",&c[i].x,&c[i].y,&c[i].r);
}
sort(c+1,c+1+n);
int cnt=n;n=0;
for(int i=1;i<cnt;i++)
for(int j=i+1;j<=cnt;j++){
if(dis(i,j)+c[i].r<=c[j].r){
cover[i]=1;
break;
}
}
for(int i=1;i<=cnt;i++)
if(!cover[i])
c[++n]=c[i];
cnt=0;
double ans=0.0;
for(int i=1;i<=n;i++){
k[++cnt].y=c[i].x-c[i].r; k[cnt].k=1;
k[++cnt].y=c[i].x+c[i].r; k[cnt].k=0;
}
sort(k+1,k+1+cnt);
int f=0;double be;
for(int i=1;i<=cnt;i++){
if(k[i].k) { if(!(f++)) be=k[i].y; }
else if(!(--f)) ans+=asr(be,k[i].y,1e-4);
}
printf("%.3f",ans);
return 0;
}
SPOJ CIRU的更多相关文章
- SPOJ CIRU SPOJ VCIRCLE 圆的面积并问题
SPOJ VCIRCLE SPOJ CIRU 两道题都是给出若干圆 就面积并,数据规模和精度要求不同. 求圆面积并有两种常见的方法,一种是Simpson积分,另一种是几何法. 在这里给出几何方法. P ...
- SPOJ CIRU - The area of the union of circles (圆的面积并)
CIRU - The area of the union of circles no tags You are given N circles and expected to calculate t ...
- SPOJ CIRU The area of the union of circles
You are given N circles and expected to calculate the area of the union of the circles ! Input The f ...
- SPOJ CIRU The area of the union of circles (计算几何)
题意:求 m 个圆的并的面积. 析:就是一个板子题,还有要注意圆的半径为0的情况. 代码如下: #pragma comment(linker, "/STACK:1024000000,1024 ...
- SPOJ CIRU The area of the union of circles ——Simpson积分
[题目分析] 圆的面积并. 直接Simpson积分,(但是有计算几何的解法,留着flag). simpson积分,如果圆出现了不连续的情况,是很容易出事情的.(脑补一下) 但是没有什么办法,本来就是一 ...
- SPOJ 8073 The area of the union of circles(计算几何の圆并)(CIRU)
Description You are given N circles and expected to calculate the area of the union of the circles ! ...
- SPOJ 8073 The area of the union of circles (圆并入门)
Sphere Online Judge (SPOJ) - Problem CIRU [求圆并的若干种算法,圆并扩展算法]_AekdyCoin的空间_百度空间 参考AekdyCoin的圆并算法解释,根据 ...
- 【题解】CIRU - The area of the union of circles [SP8073] \ 圆的面积并 [Bzoj2178]
[题解]CIRU - The area of the union of circles [SP8073] \ 圆的面积并 [Bzoj2178] 传送门: \(\text{CIRU - The area ...
- BZOJ 2588: Spoj 10628. Count on a tree [树上主席树]
2588: Spoj 10628. Count on a tree Time Limit: 12 Sec Memory Limit: 128 MBSubmit: 5217 Solved: 1233 ...
随机推荐
- 【BZOJ1367】【Baltic2004】sequence - 可合并堆
题意: 题解: 其实这是道水题啦……只不过我没做过而已 先考虑构造不严格递增序列,考虑原序列中的一段下降区间,显然区间中的$z$全取中位数最优: 那么可以把原序列拆成很多个下降序列,从头到尾加入原序列 ...
- [剑指offer] 50. 第一个只出现一次的字符 + map,hashmap 及其区别
class Solution { public: int FirstNotRepeatingChar(string str) { map<char,int>mp; ;i<str.si ...
- tp3.1 白板不报错
今天有碰上了这种情况, 一般记忆力好把刚才改动的地方恢复一下就好了,但是今天特殊原因编辑器不小心关了,也不知道把那里改坏了,一通乱找,也找不到.汗! 没办法,提交代码几面,用git看下改变的地方,是c ...
- C#中的==和Equals
== 和 Equals 简要:==比较栈上的内容,Equals比较堆上的内容 object x = 5, y = 5; Console.WriteLine(x == y); // "==&q ...
- [SharePoint2010开发入门经典]8集成业务线数据,使用业务联通服务
本章概要: 1.了解office business application 2.理解商务联通服务(BCS),如何使用BCS构建OBA 3.通过BCS使用SharePoint和办公集成技术
- Core Dataeasy出现的错误
1.2015-08-24 15:52:17.674 Tasks[3189:144763] CoreData: error: -addPersistentStoreWithType:SQLite con ...
- HDU4763-Theme Section(KMP+二分)
Theme Section Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) To ...
- android优化 清除无效代码 UCDetector
android下优化 清除无效 未被使用的 代码 UCDetector 官方下载地址:http://www.ucdetector.org/index.html UCDetector 是 eclips ...
- python爬虫系列序
关于爬虫的了解,始于看到这篇分析从数据角度解析福州美食,和上份工作中的短暂参与. 长长短短持续近一年的时间,对其态度越来越明晰,噢原来这就是我想从事的工作. 于是想要系统学习的心理便弥散开来…… 参考 ...
- webRequest
chrome.webRequest 描述: 使用 chrome.webRequest API 监控与分析流量,还可以实时地拦截.阻止或修改请求. 可用版本: 从 Chrome 17 开始支持. 权 ...