poj3819 Coverage (求直线与圆的交占直线的百分比 )
题意:给你一条直线和若干个圆,求圆与直线相交的长度占整条直线的比例
解题思路:通过定比分点的方法求出圆与直线的交占圆的比例。
第一步:(确定投影的方向是x轴还是y轴)
(1)当直线的line.s(x, y), line.e(x, y)的line.s.x与line.e.x不同一时候,这条直线能够等同于起点为line.s.x, line.e.x;
(2)不满足(1)时(即line.s.x==line.e.x时),当直线的line.s(x, y), line.e(x, y)的line.s.y与line.e.y不同一时候。这条直线能够等同于起点为line.s.x, line.e.x;
(3)当不满足(1)以及(2)时(即line.s==line.e),这时候直线为一个点,不论什么的圆都与它没有交。圆占整条直线的比例为0;
第二步:(将圆投影到第一步得到的直线上)
求出圆在直线上的投影的范围;
第三步:
求出全部圆的并。将圆的并除以线段的长度。求圆与线段的交占线段的百分比;
#include <iostream>
#include <algorithm>
#include <cstdio>
#include <cstdlib>
#include <cmath>
using namespace std;
const int MAX = 300;
struct Node
{
double x, y;
};
struct Line
{
Node s, e;
};
Line line;
Node p[MAX];
double A, B, C, delta;
double x, y, r;
double x11, y11, dx, dy;
Node tmp, cir;
double sqr(double x)
{
return x * x;
}
int circle_cross_line(Node s, Node e, Node O, double r)//推断圆与直线是否有交点
{
double x0 = O.x, y0=O.y;
x11 = s.x, y11 = s.y;
double x2 = e.x, y2 = e.y;
dx = x2 - x11, dy = y2 - y11;
A = dx * dx + dy*dy;
B = 2 * dx * (x11 - x0) + 2 * dy * (y11 - y0);
C = sqr(x11-x0) + sqr(y11-y0) - sqr(r);
delta = sqr(B) - 4 * A * C;
return delta > 0;
}
int cmp(Node a, Node b)
{
if (a.x < b.x)
return 1;
return 0;
}
int main()
{
int n, i, cnt;
int flag, flagnum;
double leng;
while (scanf("%d", &n) && n)
{
flagnum = 0;
scanf("%lf%lf%lf%lf", &line.s.x, &line.s.y, &line.e.x, &line.e.y);
if (line.s.x!=line.e.x)
{
if (line.s.x < line.e.x)
{
tmp.x = line.s.x;
tmp.y = line.e.x;
}
else
{
tmp.x = line.e.x;
tmp.y = line.s.x;
}
flag = 0;
leng = fabs(line.e.x - line.s.x);
}
else if (line.s.x==line.e.x && line.s.y!=line.e.y)
{
if (line.s.y < line.e.y)
{
tmp.x = line.s.y;
tmp.y = line.e.y;
}
else
{
tmp.x = line.e.y;
tmp.y = line.s.y;
}
flag = 1;
leng = fabs(line.e.y - line.s.y);
}
else
flagnum = 1;
cnt = 0;
for (i=0; i<n; i++)
{
scanf("%lf%lf%lf", &cir.x, &cir.y, &r);
if (flagnum)
continue;
if (circle_cross_line(line.s, line.e, cir, r))
{
p[cnt].x = (-B-sqrt(delta))/(2.0*A);
p[cnt].y = (-B+sqrt(delta))/(2.0*A);
if (flag==0)
{
p[cnt].x = p[cnt].x * dx + x11;
p[cnt].y = p[cnt].y * dx + x11;
}
else
{
p[cnt].x = p[cnt].x * dy + y11;
p[cnt].y = p[cnt].y * dy + y11;
}
if (p[cnt].x>p[cnt].y)
{
double t = p[cnt].x;
p[cnt].x = p[cnt].y;
p[cnt].y = t;
}
if (p[cnt].x>tmp.y || p[cnt].y<tmp.x)
continue;
if (p[cnt].x<tmp.x)
p[cnt].x = tmp.x;
if (p[cnt].y>tmp.y)
p[cnt].y = tmp.y;
cnt++;
}
}
if (flagnum || cnt==0)
printf("0.00\n");
else
{
sort(p, p+cnt, cmp);
double sum = 0;
tmp = p[0];
for (i=1; i<cnt; i++)
{
if (p[i].x < tmp.y)
{
if (p[i].y > tmp.y)
tmp.y = p[i].y;
}
else
{
sum += tmp.y - tmp.x;
tmp = p[i];
}
}
sum += tmp.y - tmp.x;
printf("%.2f\n", sum/leng*100.0);
}
}
return 0;
}
poj3819 Coverage (求直线与圆的交占直线的百分比 )的更多相关文章
- poj3675 求多边形与圆的面积交
题意:给出多边形的顶点坐标.圆的圆心坐标和半径,求面积交 sol:又是模板题啦= = 注意poj的C++好像认不出hypot函数,要稍微改写一下. hypot(double x,double y):即 ...
- 简单几何(直线与圆的交点) ZOJ Collision 3728
题目传送门 题意:有两个一大一小的同心圆,圆心在原点,大圆外有一小圆,其圆心有一个速度(vx, vy),如果碰到了小圆会反弹,问该圆在大圆内运动的时间 分析:将圆外的小圆看成一个点,判断该直线与同心圆 ...
- HDU 3467 (求五个圆相交面积) Song of the Siren
还没开始写题解我就已经内牛满面了,从晚饭搞到现在,WA得我都快哭了呢 题意: 在DotA中,你现在1V5,但是你的英雄有一个半径为r的眩晕技能,已知敌方五个英雄的坐标,问能否将该技能投放到一个合适的位 ...
- Agent J(求三个圆围成的区域面积)
A - A Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%lld & %llu Submit Status P ...
- hough变换是如何检测出直线和圆的?
(I)直线篇 1 直线是如何表示的?对于平面中的一条直线,在笛卡尔坐标系中,常见的有点斜式,两点式两种表示方法.然而在hough变换中,考虑的是另外一种表示方式:使用(r,theta)来表示一条直线. ...
- hdu 4404 Worms(多边形与圆的交)
求出爆炸点的坐标,就成了多边形与圆相交面积的模板题了... #include<algorithm> #include<iostream> #include<cstring ...
- zznuoj 1540 : 直线与圆
题目描述 给出一个圆的圆心坐标与圆的半径,和一条直线上的两点坐标,求这条直线与圆有多少个交点. 输入 输入3个实数x,y,r值分别表示圆心坐标与圆的半径,输入4个实数x1,y1,x2,y2表示直线上的 ...
- 【python+opencv】直线检测+圆检测
Python+OpenCV图像处理—— 直线检测 直线检测理论知识: 1.霍夫变换(Hough Transform) 霍夫变换是图像处理中从图像中识别几何形状的基本方法之一,应用很广泛,也有很多改进 ...
- hdu 3264 圆的交+二分
Open-air shopping malls Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/ ...
随机推荐
- ubuntu之安装java浏览器插件
最近搞什么openstack,在浏览器访问远程虚拟机的时候,需要浏览器有支持java.这个之前真没注意过呢, 通过自己的实践写点东西,方便一下你们搞: 1,首先去http://www.java.com ...
- JavaScript基础知识----基本语法
JavaScript 语句 JavaScript 语句向浏览器发出的命令.语句的作用是告诉浏览器该做什么. 分号 ; 分号用于分隔 JavaScript 语句. 通常我们在每条可执行的语句结尾添加分号 ...
- Iterator(迭代器)接口 --对象循环遍历
<?php class MyIterator implements Iterator { private $var = array(); public function __construct ...
- QT模态对话框用法(在UI文件中设置Widget背景图,这个图是一个带阴影边框的图片——酷)
QT弹出模态对话框做法: 1.新建UI文件时,一定要选择基类是QDialog的,我的选择是:Dialog without Buttons(),如下图: 2.然后在使用的时候: MyDialog dlg ...
- Qt打开外部程序和文件夹需要注意的细节(Qt调用VC写的动态库,VC需要用C的方式输出函数,否则MinGW32编译过程会报错)
下午写程序中遇到几个小细节,需要在这里记录一下. ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 QProcess *process = new QProcess(this ...
- cocos2d-x游戏开发系列教程-超级玛丽09-怪物激活与移动
在游戏中,很多怪物本身是会移动的,这里主要有蘑菇怪,乌龟等. 说起怪物的移动,首先在游戏里先要考虑怪物的抽象和设计. 在CMMonster.h中,有个类CMMonsterBasic,这个类抽象了所有的 ...
- Oracle笔记之对象权限与系统权限总结
对象权限与系统权限 创建表和创建session是系统权限: 系统管理员是有权限去訪问其它表的 以sys登录 sqlplus sys/on_change_install as sysdba; 创 ...
- Android基调(十六)- Service:startService()、stopService()、bindService()、unbindService()加
直行 第一 另外一种 第三种 总结 开门见山 开启服务有三种情况:假设直接使用服务,则没有必要进行绑定,可是假设要使用服务里面的方法.则要进行绑定. 具体的启动情况有下: ①调用startServic ...
- 如何优化cocos2d/x内存使用和程序大小的程序
从最初的:http://www.himigame.com/iphone-cocos2d/1043.html 译者: 在我完毕第一个游戏项目的时候.我深切地意识到"使用cocos2d来制作游戏 ...
- Oracle统计之like,or关键字
有的时候我们统计需要like多个条件,比如某个字段中含有 “银行”或者“投资公司”,这个时候就用like多个字段. 如下是统计某个地区的2015年入院的病种诊断中带有 “癌”或者“肿瘤”的情况, 正确 ...