ZOJ 2589 Circles(平面图欧拉公式)
【题目链接】 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=2589
【题目大意】
给出一些圆,问这些圆可以把平面分为几个部分。
【题解】
我们发现圆交图一定是个平面图,因此可以用平面图欧拉公式R=E-V+2
但是我们发现有些圆并不相交,因此每个图需要单独完全计算,
我们计算每个封闭图形的平面数,他们的和+1便是答案,
考虑单独的封闭图形有R=E-V+1,在下图中:
我们发现当蓝色的圆加入图中之后,他为平面增加的点数是4,增加的边数是8,
其中属于蓝色的圆弧的边数为4,其余四条增加的边源于红色和黄色圆弧上点的增加,
所以我们发现对于一个圆来说,它为平面贡献的边数为其与其余圆的交点数,
至于封闭平面图形点的计算,我们在搜索中用set来去重即可。
【代码】
#include <cstdio>
#include <algorithm>
#include <cmath>
#include <cstring>
#include <vector>
#include <set>
using namespace std;
double eps=1e-8;
int sgn(double x) {
if(x<-eps)return -1;
if(x>eps)return 1;
return 0;
}
struct vec{
double x,y;
vec(double x=0,double y=0):x(x),y(y){}
vec operator + (vec v){return vec(x+v.x,y+v.y);}
vec operator - (vec v){return vec(x-v.x,y-v.y);}
vec operator * (double v){return vec(x*v,y*v);}
vec operator / (double v){return vec(x/v,y/v);}
bool operator < (const vec &rhs)const{
if(sgn(x-rhs.x)!=0)return x<rhs.x;
else if(sgn(y-rhs.y)!=0)return y<rhs.y;
else return false;
}
bool operator ==(const vec &rhs)const{
return sgn(x-rhs.x)==0&&sgn(y-rhs.y)==0;
}
double operator *(vec v){return x*v.x+y*v.y;}
double len(){return hypot(x,y);}
double len_sqr(){return x*x+y*y;}
double angle(){return atan2(y,x);}
//逆时针旋转
vec rotate(double c){return vec(x*cos(c)-y*sin(c),x*sin(c)+y*cos(c));}
vec trunc(double l){return (*this)*l/len();}
vec rot90(){return vec(-y,x);}
};
struct circle{
vec c;double r;
circle(vec c=vec(0,0),double r=0):c(c),r(r){}
vec point(const double &a)const{
return vec(c.x+cos(a)*r,c.y+sin(a)*r);
}
};
//圆圆相交
int circle_circle_intersection(circle a,circle b,vec &p1,vec &p2) {
double d=(a.c-b.c).len();
if(sgn(d)==0)return 0;
if(sgn(a.r+b.r-d)<0||sgn(fabs(a.r-b.r)-d)>0)return false;//相离|内含
double an=(b.c-a.c).angle();
double da=acos((a.r*a.r+d*d-b.r*b.r)/(2*a.r*d));
p1=a.point(an-da);
p2=a.point(an+da);
if(p1==p2)return 1;
else return 2;
}
const int N=60;
vector<circle> cir;
vector<int> G[N];
set<vec> dfs_save,set_p[N];
set<vec>::iterator it;
int v[N],E,T,n;
void dfs(int x){
v[x]=1;
for(it=set_p[x].begin();it!=set_p[x].end();it++)dfs_save.insert(*it);
E+=(int)set_p[x].size();
for(int i=0;i<G[x].size();i++)if(!v[G[x][i]])dfs(G[x][i]);
}
int main(){
scanf("%d",&T);
while(T--){
scanf("%d",&n); cir.clear();
for(int i=0;i<n;i++)G[i].clear(),set_p[i].clear();
memset(v,0,sizeof(v));
for(int i=0;i<n;i++){
double x,y,r;
scanf("%lf%lf%lf",&x,&y,&r);
cir.push_back(circle(vec(x,y),r));
}
for(int i=0;i<n;i++)
for(int j=i+1;j<n;j++){
vec a,b;
int u=circle_circle_intersection(cir[i],cir[j],a,b);
if(u){
G[i].push_back(j); G[j].push_back(i);
set_p[i].insert(a); set_p[j].insert(a);
set_p[i].insert(b); set_p[j].insert(b);
}
}int ans=0;
for(int i=0;i<n;i++){
if(!v[i]){
dfs_save.clear(); E=0;
dfs(i); ans+=E-(int)dfs_save.size()+1;
}
}printf("%d\n",ans+1);
}return 0;
}
ZOJ 2589 Circles(平面图欧拉公式)的更多相关文章
- P7295-[USACO21JAN]Paint by Letters P【平面图欧拉公式】
正题 题目链接:https://www.luogu.com.cn/problem/P7295 题目大意 给出\(n*m\)的网格,每个格子上有字母,相同字母的四联通相邻格子为连通,每次询问一个子矩阵求 ...
- POJ--2284--That Nice Euler Circuit【平面图欧拉公式】
链接:id=2284">http://poj.org/problem?id=2284 题意:一个自己主动绘图的机器在纸上(无限大)绘图,笔尖从不离开纸,有n个指令,每一个指令是一个坐标 ...
- zoj 2589 Matrix Searching 二维线段树
题目链接 给一个n*n的矩阵, 给q个查询, 每次给出x1, y1, x2, y2, 求这个矩阵中的最小值. 代码基本上和上一题相同... #include<bits/stdc++.h> ...
- ACM计算几何题目推荐
//第一期 计算几何题的特点与做题要领: 1.大部分不会很难,少部分题目思路很巧妙 2.做计算几何题目,模板很重要,模板必须高度可靠. 3.要注意代码的组织,因为计算几何的题目很容易上两百行代码,里面 ...
- ZOJ 1608 Two Circles and a Rectangle
Give you two circles and a rectangle, your task is to judge wheather the two circles can be put into ...
- poj2284 That Nice Euler Circuit(欧拉公式)
题目链接:poj2284 That Nice Euler Circuit 欧拉公式:如果G是一个阶为n,边数为m且含有r个区域的连通平面图,则有恒等式:n-m+r=2. 欧拉公式的推广: 对于具有k( ...
- POJ2284 That Nice Euler Circuit (欧拉公式)(计算几何 线段相交问题)
That Nice Euler Circuit Time Limit: 3000MS M ...
- 【转载】图论 500题——主要为hdu/poj/zoj
转自——http://blog.csdn.net/qwe20060514/article/details/8112550 =============================以下是最小生成树+并 ...
- UVa 10213 (欧拉公式+Java大数) How Many Pieces of Land ?
题意: 一块圆形土地,在圆周上选n个点,然后两两连线,问把这块土地分成多少块? 分析: 首先紫书上的公式是错的,不过根据书上提供的思路很容易稍加修改得到正确答案! 然后推公式吧,这里用到平面图的欧拉公 ...
随机推荐
- 复现VGG19训练自定义图像分类
1.复现VGG训练自定义图像分类,成功了哈哈. 需要代码工程可联系博主qq号,在左边连接可找到. 核心代码: # coding:utf-8 import tensorflow as tf import ...
- android隐藏EditText光标
在android中如果有EditText,那么在载入时,光标会默认显示在第一个EditText框中,如果不想显示光标,且也不想把该光标移动到下一个EditText框,最简单的方法是在该 EditTex ...
- LeetCode 20 Generate Parentheses
Given n pairs of parentheses, write a function to generate all combinations of well-formed parenthes ...
- imx6设备树pinctrl解析【转】
转自:http://blog.csdn.net/michaelcao1980/article/details/50730421 版权声明:本文为博主原创文章,未经博主允许不得转载. 最近在移植linu ...
- $(document).ready 和 window.onload 的区别
1.相同点 两者都用于在网页加载完后执行相应代码块. 2.不同点 window.onload 在创建完 DOM 树后,所有外部资源(图片.Flash 动画等)加载完成,且整个页面在浏览器窗口中显示完毕 ...
- 浅析linux内核中timer定时器的生成和sofirq软中断调用流程【转】
转自:http://blog.chinaunix.net/uid-20564848-id-73480.html 浅析linux内核中timer定时器的生成和sofirq软中断调用流程 mod_time ...
- 64_j2
jetty-websocket-server-9.4.3-3.v20170317.fc26.n..> 14-Apr-2017 12:03 62034 jetty-websocket-servle ...
- Linux配置Tomcat
系统:Ubuntu,Tomcat:apache-tomcat-8.5.23.tar.gz 1,找到apache-tomcat-8.5.23.tar.gz,复制到 /usr/local root@ubu ...
- IE中部分版本的浏览器对Select标签设置innerHTML无效的问题
这样写的代码:document.getElementById('data_list').innerHTML = data;//data是ajax返回的数据 结果发现在ie10的兼容模式下面下拉框没有内 ...
- linux命令行任务管理
今天看到了linux命令行的任务管理命令感觉很实用: 1.ctrl+z 将当前前台执行的任务放到后台并暂停 2.fg恢复上次放入后台的任务 这两个命令组合起来很实用,比如在linux命令行中写pyt ...