bjfu1235 两圆公共面积
给定两个圆,求其覆盖的面积,其实也就是求其公共面积(然后用两圆面积和减去此值即得最后结果)。
我一开始是用计算几何的方法做的,结果始终不过。代码如下:
/*
* Author : ben
*/
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <ctime>
#include <iostream>
#include <algorithm>
#include <queue>
#include <set>
#include <map>
#include <stack>
#include <string>
#include <vector>
#include <deque>
#include <list>
#include <functional>
#include <numeric>
#include <cctype>
using namespace std;
const double pi = acos(-);
typedef struct MyPoint {
double x, y;
MyPoint(double xx = , double yy = ) {
x = xx;
y = yy;
}
} MyPoint; inline double mydistance2(const MyPoint &p1, const MyPoint &p2) {
return (p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y);
} inline double mydistance(const MyPoint &p1, const MyPoint &p2) {
return sqrt(mydistance2(p1, p2));
} MyPoint intersection(MyPoint u1, MyPoint u2, MyPoint v1, MyPoint v2) {
MyPoint ret = u1;
double t = ((u1.x - v1.x) * (v1.y - v2.y) - (u1.y - v1.y) * (v1.x - v2.x))
/ ((u1.x - u2.x) * (v1.y - v2.y) - (u1.y - u2.y) * (v1.x - v2.x));
ret.x += (u2.x - u1.x) * t;
ret.y += (u2.y - u1.y) * t;
return ret;
} void intersection_line_circle(MyPoint c, double r, MyPoint l1, MyPoint l2,
MyPoint& p1, MyPoint& p2) {
MyPoint p = c;
double t;
p.x += l1.y - l2.y;
p.y += l2.x - l1.x;
p = intersection(p, c, l1, l2);
t = sqrt(r * r - mydistance(p, c) * mydistance(p, c)) / mydistance(l1, l2);
p1.x = p.x + (l2.x - l1.x) * t;
p1.y = p.y + (l2.y - l1.y) * t;
p2.x = p.x - (l2.x - l1.x) * t;
p2.y = p.y - (l2.y - l1.y) * t;
} void intersection_circle_circle(MyPoint c1, double r1, MyPoint c2, double r2,
MyPoint& p1, MyPoint& p2) {
MyPoint u, v;
double t;
t = ( + (r1 * r1 - r2 * r2) / mydistance(c1, c2) / mydistance(c1, c2)) / ;
u.x = c1.x + (c2.x - c1.x) * t;
u.y = c1.y + (c2.y - c1.y) * t;
v.x = u.x + c1.y - c2.y;
v.y = u.y - c1.x + c2.x;
intersection_line_circle(c1, r1, u, v, p1, p2);
} int main() {
freopen("data.in", "r", stdin);
// freopen("data.out", "w", stdout);
int T;
double x, y, r1, r2;
scanf("%d", &T);
double ans;
while (T--) {
scanf("%lf%lf%lf", &x, &y, &r1);
MyPoint c1(x, y);
scanf("%lf%lf%lf", &x, &y, &r2);
MyPoint c2(x, y);
double dis2 = mydistance2(c1, c2);
double dis = sqrt(dis2);
if (dis >= r1 + r2) { //相离
ans = pi * r1 * r1 + pi * r2 * r2;
} else if (dis <= fabs(r1 - r2)) { //包含
double r = r1 > r2 ? r1 : r2;
ans = pi * r * r;
} else { //相交
MyPoint p1, p2;
intersection_circle_circle(c1, r1, c2, r2, p1, p2);
double d2 = mydistance(p1, p2) / ;
double angle1 = asin(d2 / r1);
double angle2 = asin(d2 / r2);
double Sanjiao1 = sqrt(r1 * r1 - d2 * d2) * d2;
double Sanjiao2 = sqrt(r2 * r2 - d2 * d2) * d2;
double San1 = r1 * r1 * angle1;
double San2 = r2 * r2 * angle2;
ans = pi * r1 * r1 + pi * r2 * r2;
ans -= San1 + San2 - Sanjiao1 - Sanjiao2;
}
printf("%.6f\n", ans);
}
return ;
}
根据后来的调试,应该是对如下图b的情况处理不正确。


于是后来上网找了几个中学的解析几何公式,终于a了。
做法是联立两个圆的方程(相减),得到相交弦所在直线方程,然后用点到直接的距离公式得到h1和h2,接着算出θ1和θ2,然后就能求得三角形的面积和扇形的面积了。一开始我以为需要分类讨论上面图a和图b两种情况,后来发现,直接去掉求距离时的取绝对值运算就可以了,因为距离为负的时候,得到的夹角也是负的,这样求的三角形面积是负的,扇形也是原先的相补的那部分,具体的图我就不画了,很容易想明白的。
AC代码如下:
/*
* Author : ben
*/
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <ctime>
#include <iostream>
#include <algorithm>
#include <queue>
#include <set>
#include <map>
#include <stack>
#include <string>
#include <vector>
#include <deque>
#include <list>
#include <functional>
#include <numeric>
#include <cctype>
using namespace std;
const double pi = acos(-);
typedef struct MyPoint {
double x, y;
MyPoint(double xx = , double yy = ) {
x = xx;
y = yy;
}
} MyPoint; inline double mydistance2(const MyPoint &p1, const MyPoint &p2) {
return (p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y);
} int main() {
// freopen("data.in", "r", stdin);
int T;
scanf("%d", &T);
double ans, r1, r2;
MyPoint c1, c2;
while (T--) {
scanf("%lf%lf%lf", &c1.x, &c1.y, &r1);
scanf("%lf%lf%lf", &c2.x, &c2.y, &r2);
double dis2 = mydistance2(c1, c2);
double dis = sqrt(dis2);
if (dis >= r1 + r2) { //相离
ans = pi * r1 * r1 + pi * r2 * r2;
} else if (dis <= fabs(r1 - r2)) { //包含
double r = r1 > r2 ? r1 : r2;
ans = pi * r * r;
} else { //相交
//h1和h2可能为负
double h1 = (dis2 + r1 * r1 - r2 * r2) / dis / 2.0;
double h2 = dis - h1;
double angle1 = acos(h1 / r1);
double angle2 = acos(h2 / r2);
double Sanjiao = sqrt(r1 * r1 - h1 * h1) * dis;
double Sanxin1 = r1 * r1 * angle1;
double Sanxin2 = r2 * r2 * angle2;
ans = pi * r1 * r1 + pi * r2 * r2;
ans -= Sanxin1 + Sanxin2 - Sanjiao;
}
printf("%.6f\n", ans);
}
return ;
}
bjfu1235 两圆公共面积的更多相关文章
- poj 2546(两圆公共面积)
Circular Area Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 5682 Accepted: 2225 Des ...
- poj2546Circular Area(两圆相交面积)
链接 画图推公式 这两种情况 都可用一种公式算出来 就是两圆都求出圆心角 求出扇形的面积减掉三角形面积 #include <iostream> using namespace std; # ...
- [hdu 3264] Open-air shopping malls(二分+两圆相交面积)
题目大意是:先给你一些圆,你可以任选这些圆中的一个圆点作圆,这个圆的要求是:你画完以后.这个圆要可以覆盖之前给出的每一个圆一半以上的面积,即覆盖1/2以上每一个圆的面积. 比如例子数据,选左边还是选右 ...
- hdu5858 Hard problem(求两圆相交面积)
题目传送门 Hard problem Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Other ...
- hdu 3264(枚举+二分+圆的公共面积)
Open-air shopping malls Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/ ...
- codeforce gym/100495/problem/K—Wolf and sheep 两圆求相交面积 与 gym/100495/problem/E—Simple sequence思路简述
之前几乎没写过什么这种几何的计算题.在众多大佬的博客下终于记起来了当时的公式.嘚赶快补计算几何和概率论的坑了... 这题的要求,在对两圆相交的板子略做修改后,很容易实现.这里直接给出代码.重点的部分有 ...
- POJ 2546 & ZOJ 1597 Circular Area(求两圆相交的面积 模板)
题目链接: POJ:http://poj.org/problem? id=2546 ZOJ:problemId=597" target="_blank">http: ...
- 求两圆相交部分面积(C++)
已知两圆圆心坐标和半径,求相交部分面积: #include <iostream> using namespace std; #include<cmath> #include&l ...
- 简单几何(圆与多边形公共面积) UVALive 7072 Signal Interference (14广州D)
题目传送门 题意:一个多边形,A点和B点,满足PB <= k * PA的P的范围与多边形的公共面积. 分析:这是个阿波罗尼斯圆.既然是圆,那么设圆的一般方程:(x + D/2) ^ 2 + (y ...
随机推荐
- hdu 1850 Being a Good Boy in Spring Festival 博弈论
求可行的方案数!! 代码如下: #include<stdio.h> ]; int main(){ int n,m; while(scanf("%d",&n)&a ...
- 深入浅出Java并发包—原子类操作
我们知道,JDK1.5以后引入了并发包(java.util.concurrent)用于解决多CPU时代的并发问题,而并发包中的类大部分是基于Queue的并发类,Queue在大多数情况下使用了原子类(A ...
- Java 编译错误:缺少返回语句
示例: import java.util.*; import java.io.*; public class tt { public static void main(String[] args) { ...
- React用JS 模拟动画介绍
一. <!DOCTYPE html> <html lang="zh-cn"> <head> <meta charset="UTF ...
- Delphi判断字符串是否是数字、字母、大小写字母
function IsNumberic(Vaule:String):Boolean; //判断Vaule是不是数字 var i:integer; begin result:=true; //设置返回值 ...
- Android 虚线分割Shape
<?xml version="1.0" encoding="utf-8"?> <shape xmlns:android="http: ...
- 使用XSSFWork创建的xlsx后缀Excel文件无法打开
使用XSSFWork创建的xlsx后缀Excel文件无法打开 标签: POIExcelmicrosoftxlsx 2015-04-21 10:49 1170人阅读 评论(2) 收藏 举报 分类: Ja ...
- Linux使用者管理(2)---账号管理
用户添加 新增用户 sudo useradd -m username 这里必须使用sudo 因为需要对/etc/shadow进行读写,在ubuntu环境下,必须使用-m设置,否则不会创建主文件夹. 在 ...
- Java API —— System类
1.System类概述 System 类包含一些有用的类字段和方法.它不能被实例化. 2.成员方法 public static void gc():运行垃圾回收器 ...
- mysql 行列动态转换(列联表,交叉表)
mysql 行列动态转换(列联表,交叉表) (1)动态,适用于列不确定情况 create table table_name( id int primary key, col1 char(2), col ...