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;
}
时间: 2024-12-08 11:49:37

poj3819 Coverage (求直线与圆的交占直线的百分比 )的相关文章

hdu 3264 圆的交+二分

Open-air shopping malls Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 1822    Accepted Submission(s): 651 Problem Description The city of M is a famous shopping city and its open-air shopping

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

题目传送门 题意:有两个一大一小的同心圆,圆心在原点,大圆外有一小圆,其圆心有一个速度(vx, vy),如果碰到了小圆会反弹,问该圆在大圆内运动的时间 分析:将圆外的小圆看成一个点,判断该直线与同心圆的交点,根据交点个数计算时间.用到了直线的定义,圆的定义,直线与圆交点的个数. /************************************************ * Author :Running_Time * Created Time :2015/10/24 星期六 16:14:

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

还没开始写题解我就已经内牛满面了,从晚饭搞到现在,WA得我都快哭了呢 题意: 在DotA中,你现在1V5,但是你的英雄有一个半径为r的眩晕技能,已知敌方五个英雄的坐标,问能否将该技能投放到一个合适的位置,使得对面所有敌人都被眩晕,这样你就有机会能够逃脱. 分析: 对于敌方一个英雄来说,如果技能的投放位置距离他不超过r则满足要求,那么如果要眩晕所有的敌人,可行区域就是以五人为中心的半径为r的圆的相交区域. 现在问题就转化为求五个半径相同的圆的相交部分的面积,如果只有一个点则输出该点. 在求交之前,

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

(I)直线篇 1 直线是如何表示的?对于平面中的一条直线,在笛卡尔坐标系中,常见的有点斜式,两点式两种表示方法.然而在hough变换中,考虑的是另外一种表示方式:使用(r,theta)来表示一条直线.其中r为该直线到原点的距离,theta为该直线的垂线与x轴的夹角.如下图所示. 2 如果坐标系中有多个点,又怎样识别出哪些点在一条直线上呢?使用hough变换来检测直线的思想就是:为每一个点假设n个方向的直线,通常n=180,此时检测的直线的角度精度为1°,分别计算这n条直线的(r,theta)坐标

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

A - A Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%lld & %llu Submit Status Practice LightOJ 1331 Description Agent J is preparing to steal an antique diamond piece from a museum. As it is fully guarded and they are guarding it usi

OpenCV2学习笔记(八):使用霍夫变换检测直线和圆

在研究一幅图像时,常常会遇到一些平面或线性问题,直线在图像中频繁可见.这些富有意义的特征在物体识别等图像处理过程中扮演着重要的角色.本节主要记录一种经典的检测直线算法--霍夫变换(Hough Transform),用Hough变换检测图像中的直线和圆,开发平台为Qt5.3.2+OpenCV2.4.9. 一:Hough变换检测图像的直线 1.基础Hough变换 在霍夫变换中,直线用以下方程表示: 其中,参数表示一条直线到图像原点(左上角)的距离, 表示与直线垂直的角度.如下图所示,直线1的垂直线的

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

(I)直线篇 1 直线是如何表示的? 对于平面中的一条直线,在笛卡尔坐标系中,常见的有点斜式,两点式两种表示方法.然而在hough变换中,考虑的是另外一种表示方式:使用(r,theta)来表示一条直线.其中r为该直线到原点的距离,theta为该直线的垂线与x轴的夹角.如下图所示. 2 如果坐标系中有多个点,又怎样识别出哪些点在一条直线上呢? 使用hough变换来检测直线的思想就是:为每一个点假设n个方向的直线,通常n=180,此时检测的直线的角度精度为1°,分别计算这n条直线的(r,theta)

直线与圆

前言 当涉及直线与圆的相关问题时,考查最多的知识点是其中的\(Rt\triangle\).注:其中的\(Rt\triangle\)指半弦长与半径和弦心距构成的直角三角形. 典例剖析 例1[2019届高三理科数学第三轮模拟训练题]直线\(l:kx-2y-3=0\)与圆\(C:(x-1)^2+(y+2)^2=4\)交于\(A\),\(B\)两点,若\(\triangle ABC\)的周长为\(4+2\sqrt{3}\),则实数\(k\)的值为[] $A.\cfrac{3}{2}$ $B.-\cfra

Python下opencv使用笔记(十一)(详解hough变换检测直线与圆)

在数字图像中,往往存在着一些特殊形状的几何图形,像检测马路边一条直线,检测人眼的圆形等等,有时我们需要把这些特定图形检测出来,hough变换就是这样一种检测的工具. Hough变换的原理是将特定图形上的点变换到一组参数空间上,根据参数空间点的累计结果找到一个极大值对应的解,那么这个解就对应着要寻找的几何形状的参数(比如说直线,那么就会得到直线的斜率k与常熟b,圆就会得到圆心与半径等等). 关于hough变换,核心以及难点就是关于就是有原始空间到参数空间的变换上.以直线检测为例,假设有一条直线L,