题意:给你一条直线和若干个圆,求圆与直线相交的长度占整条直线的比例

解题思路:通过定比分点的方法求出圆与直线的交占圆的比例。

第一步:(确定投影的方向是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 (求直线与圆的交占直线的百分比 )的更多相关文章

  1. poj3675 求多边形与圆的面积交

    题意:给出多边形的顶点坐标.圆的圆心坐标和半径,求面积交 sol:又是模板题啦= = 注意poj的C++好像认不出hypot函数,要稍微改写一下. hypot(double x,double y):即 ...

  2. 简单几何(直线与圆的交点) ZOJ Collision 3728

    题目传送门 题意:有两个一大一小的同心圆,圆心在原点,大圆外有一小圆,其圆心有一个速度(vx, vy),如果碰到了小圆会反弹,问该圆在大圆内运动的时间 分析:将圆外的小圆看成一个点,判断该直线与同心圆 ...

  3. HDU 3467 (求五个圆相交面积) Song of the Siren

    还没开始写题解我就已经内牛满面了,从晚饭搞到现在,WA得我都快哭了呢 题意: 在DotA中,你现在1V5,但是你的英雄有一个半径为r的眩晕技能,已知敌方五个英雄的坐标,问能否将该技能投放到一个合适的位 ...

  4. Agent J(求三个圆围成的区域面积)

    A - A Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%lld & %llu Submit Status P ...

  5. hough变换是如何检测出直线和圆的?

    (I)直线篇 1 直线是如何表示的?对于平面中的一条直线,在笛卡尔坐标系中,常见的有点斜式,两点式两种表示方法.然而在hough变换中,考虑的是另外一种表示方式:使用(r,theta)来表示一条直线. ...

  6. hdu 4404 Worms(多边形与圆的交)

    求出爆炸点的坐标,就成了多边形与圆相交面积的模板题了... #include<algorithm> #include<iostream> #include<cstring ...

  7. zznuoj 1540 : 直线与圆

    题目描述 给出一个圆的圆心坐标与圆的半径,和一条直线上的两点坐标,求这条直线与圆有多少个交点. 输入 输入3个实数x,y,r值分别表示圆心坐标与圆的半径,输入4个实数x1,y1,x2,y2表示直线上的 ...

  8. 【python+opencv】直线检测+圆检测

     Python+OpenCV图像处理—— 直线检测 直线检测理论知识: 1.霍夫变换(Hough Transform) 霍夫变换是图像处理中从图像中识别几何形状的基本方法之一,应用很广泛,也有很多改进 ...

  9. hdu 3264 圆的交+二分

    Open-air shopping malls Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/ ...

随机推荐

  1. python成长之路9——socket和socketserver

    IPC:进程间通信 本地的进程间通信(IPC)有很多种方式,但可以总结为下面4类: 消息传递(管道.FIFO.消息队列) 同步(互斥量.条件变量.读写锁.文件和写记录锁.信号量) 共享内存(匿名的和具 ...

  2. [LeetCode]题解(python):025-Reverse Nodes in k-Group

    题目来源: https://leetcode.com/problems/reverse-nodes-in-k-group/ 题意分析: 这道题目和上一题目类似,输入一个链表和一个整型k.每k个翻转一下 ...

  3. 开启Linux VNC远程桌面

    Xwindows:gnome (红帽默认安装的图形界面)   一, 确认及安装VNCSERVER. 1,首先确认你服务器是否配置了VNCSERVER,可以在命令行下敲入以下命令查看: [root@lo ...

  4. wiki oi 3115高精度练习之减法

    题目描述 Description 给出两个正整数A和B,计算A-B的值.保证A和B的位数不超过500位. 输入描述 Input Description 读入两个用空格隔开的正整数 输出描述 Outpu ...

  5. C++一些注意点之异常处理

    几篇文章:(1)http://blog.csdn.net/daheiantian/article/details/6530318 (2)http://blog.chinaunix.net/uid-21 ...

  6. Co-prime(容斥)

    Co-prime Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total S ...

  7. VMware-WorkStation 去掉VM工具栏

    链接地址:http://jingyan.baidu.com/article/8275fc86b95c6246a03cf629.html VMware 是目前用得最多的虚拟机.当开启虚拟机的全屏模式时, ...

  8. 使用ownCloud在Linux安装你的个人云服务

    ownCloud是一个免费开源的软件,用于为分享文件,日历,联系人,书签和个人音频/视频.非常容易安装和管理. 前提 在这篇教程里我使用CentOS 6.5 minimal server来安装ownC ...

  9. python中enumerate 函数

    enumerate 函数用于遍历序列中的元素以及它们的下标: >>> for i,j in enumerate(('a','b','c')):  print i,j 0 a 1 b ...

  10. 开大Stack的一个小技巧

    在程序头部添加一行 #pragma comment(linker, "/STACK:16777216") 可有效开大堆栈 实验效果如下: 11330179 2014-08-05 1 ...