判断两条线段是否相交:

矢量

如果一条线段的端点是有次序之分的话,那么这种线段就称为 有向线段,如果有向线段p1p2的起点p1在坐标的原点,则可以把它称为矢量p2

矢量的加减

设二维矢量 P = (x1, y1), Q = (x2, y2),则P + Q = (x1 + x2, y1 + y2), P - Q = (x1 - x2, y1 - y2),且有P + Q = Q + P, P - Q = -(Q - P)

矢量叉积

设矢量 P = (x1, y1), Q = (x2, y2),则P * Q = x1 * y2 - x2 * y1;其结果是一个由(0, 0), P, Q, P + Q所组成的平行四边形的 带符号的面积,P * Q = -(Q * P), P * (- Q) = -(P * Q)

叉积的一个非常重要的性质是可以通过它的符号来判断两矢量相互之间的顺逆时针关系:

若 P * Q > 0,则 P 在 Q的顺时针方向

若 P * Q < 0, 则 P 在 Q的逆时针方向

若 P * Q = 0,则 P 与 Q共线,但不确定P, Q的方向是否相同

折线段的拐向判断

如图,假设有折线段p0p1p2,要确定p1p2相对于p0p1是向左拐还是向右拐,可以通过计算(p2 - p0)*(p1 - p0)的符号来确定折线的拐向(点p2 - p0实际上就是向量p2,但这里要注意线段和矢量的区别)

判断点是否在线段上

设点 Q = (x, y), P1 = (x1, y1), P2 = (x2, y2),若(Q - P1)*(P2 - P1) = 0且min(x1, x2) <= x <= max(x1, x2) && min(y1, y2) <= y <= max(y1, y2),则点Q在线段P1P2上

判断两线段是否相交

1)快速排斥试验

设以线段 P1P2 为对角线的矩形为R,设以线段Q1Q2为对角线的矩形为T,若R、T不相交,则两线段不可能相交

假设 P1 = (x1, y1), P2 = (x2, y2), Q1 = (x3, y3), Q2 = (x4, y4),设矩形R的x 坐标的最小边界为minRX = min(x1, x2),以此类推,将矩形表示为R = (minRX, minRY, maxRX, maxRY)的形式,若两矩形相交,则相交的部分构成了一个新的矩形F,如图所示,我们可以知道F的minFX = max(minRX, minTX), minFY = max(minRY, minTY), maxFX = min(maxRX, maxTX), maxFY = min(maxRY, maxTX),得到F的各个值之后,只要判断矩形F是否成立就知道R和T 到底有没有相交了,若minFX > maxFX或minFY > maxFy则F无法构成,RT不相交,否则RT相交

 
 
2)跨立试验

若 P1P2 跨立 Q1Q2,则 P1,P2 分别在 Q1Q2 所在直线的两端,则有(P1 - Q1)*(Q2 - Q1) * (Q2 - Q1)*(P2 - Q1) > 0,当(P1 - Q1)*(Q2 - Q1) = 0 时,说明(P1 - Q1)与(Q2 - Q1)共线,但由于已经经过快速排斥试验,所以Q1必为P1P2与 Q1Q2的交点,依然满足线段相交的条件,则跨立试验可改为:

当 (P1 - Q1)*(Q2 - Q1) * (Q2 - Q1)*(P2 - Q1) >= 0,则P1P2跨立Q1Q2,当Q1Q2也跨立P1P2的时候,则P1P2相交

(注意式子中被隔开的*代表两个叉积的值的相乘,而另外的两个*则代表计算矢量叉积)

判断线段与直线是否相交:

首先在直线上随便取两点做为线段l1;

再与线段L2进行判断,因为是直线所以现在只需判断L2是否跨立l1就行了。

代码如下:

bool inter_line(line l1,line l2)//判断直线与线段是否相交

{

return sgn((l2.s-l1.e)^(l1.s-l1.e))*sgn((l2.e-l1.e)^(l1.s-l1.e))<=0;

}

代码实现:

再这之前可以先去学下,C++的运算符重载部分

运算符重载详解  http://blog.chinaunix.NET/uid-21411227-id-1826759.html

#include <iostream>

#include <cstdio>

#include <cmath>

using namespace std;

const double eps=1e-8;

const double PI=acos(-1.0);

int sgn(double x)

{

if(fabs(x)<eps) return 0;

if(x < 0) return -1;

else return 1;

}

struct point

{

double x,y;

point(){}

point (double _x,double _y)

{

x=_x,y=_y;

}

point operator -(const point &b)const

{

return point(x-b.x,y-b.y);

}

point operator +(const point &b)const

{

return point (x+b.x,y+b.y);

}

double operator ^(const point &b)const //叉积

{

return x*b.y-y*b.x;

}

double operator *(const point &b)const //点积

{

return x*b.x+y*b.y;

}

void transxy(double B)  //绕原点旋转角度B(弧度值),后的x,y

{

double tx=x,ty=y;

x= tx*cos(B) - ty*sin(B);

y= tx*sin(B) + ty*cos(B);

}

};

struct line//线

{

point s,e;

line(){}

line(point _s,point _e)

{

s=_s,e=_e;

}

};

double dist(point a,point b)//两点间距离

{

return sqrt((a-b)*(a-b));

}

bool inter(line l1,line l2)//判断线段相交

{

return max(l1.s.x,l1.e.x)>=min(l2.s.x,l2.e.x)&&

max(l2.s.x,l2.e.x)>=min(l1.s.x,l1.e.x)&&

max(l1.s.y,l1.e.y)>=min(l2.s.y,l2.e.y)&&

max(l2.s.y,l2.e.y)>=min(l1.s.y,l1.e.y)&&

sgn((l2.s-l1.e)^(l1.s-l1.e))*sgn((l2.e-l1.e)^(l1.s-l1.e))<=0&&

sgn((l1.s-l2.e)^(l2.s-l2.e))*sgn((l1.e-l2.e)^(l2.s-l2.e))<=0;

}

bool inter_line(line l1,line l2)//判断直线与线段是否相交

{

return sgn((l2.s-l1.e)^(l1.s-l1.e))*sgn((l2.e-l1.e)^(l1.s-l1.e))<=0;

}

int main()

{

double ax,ay,bx,by,cx,cy,dx,dy;

while(scanf("%lf%lf%lf%lf%lf%lf%lf%lf",&ax,&ay,&bx,&by,&cx,&cy,&dx,&dy)!=EOF)//分别输入两条线段的首尾端点

{

point k1(ax,ay);

point k2(bx,by);

point k3(cx,cy);

point k4(dx,dy);

line l1(k1,k2);

line l2(k3,k4);

if(inter(l1,l2))

printf("Yes\n");

else

printf("No\n");

}

return 0;

}

转载acm几何基础(2)的更多相关文章

  1. [转载]ACM(访问控制模型),Security Identifiers(SID),Security Descriptors(安全描述符),ACL(访问控制列表),Access Tokens(访问令牌)

    对于<windows核心编程>中的只言片语无法驱散心中的疑惑.就让MSDN中的解释给我们一盏明灯吧.如果要很详细的介绍,还是到MSDN仔细的看吧,我只是大体用容易理解的语言描述一下. wi ...

  2. 转载 ACM训练计划

    leetcode代码 利用堆栈:http://oj.leetcode.com/problems/evaluate-reverse-polish-notation/http://oj.leetcode. ...

  3. [转载]ACM搜索算法总结(总结)

    原文地址:ACM搜索算法总结(总结)作者:GreenHand 搜索是ACM竞赛中的常见算法,本文的主要内容就是分析它的 特点,以及在实际问题中如何合理的选择搜索方法,提高效率.文章的第一部分首先分析了 ...

  4. 转载- ACM常见的各种说法

    from : http://blog.csdn.net/qq_15015129/article/details/52738184 1.答案错误 —— wrong answer 就是最常见的.这个没办法 ...

  5. 【转载】 从ACM会议看中国大陆计算机科学与国外的差距

    ps:   这是一篇06年的文章,与今日的国内计算机行业学术圈环境简直是天翻地覆,很不错的history,值得mark下,今日的cs学术发展十号是坏不发表意见,但是history是值得对比,借鉴,思考 ...

  6. ACM之路(转载)

    转载自:https://www.cnblogs.com/tianjintou/p/4139416.html 要注意,ACM的竞赛性强,因此自己应该和自己的实际应用联系起来. 适合自己的才是好的,有的人 ...

  7. 【转载】ACM总结

    转自亲学长的总结 声明:本文是写给弱校ACM新手的一点总结,受自身水平和眼界所限,难免会有一些个人主观色彩,希望大牛指正 感谢@Wackysoft .@哇晴天 . @ 一切皆有可能1 的指教,现根据这 ...

  8. 北大ACM题库习题分类与简介(转载)

    在百度文库上找到的,不知是哪位大牛整理的,真的很不错! zz题 目分类 Posted by fishhead at 2007-01-13 12:44:58.0 -------------------- ...

  9. 计算几何基础——矢量和叉积 && 叉积、线段相交判断、凸包(转载)

    转载自 http://blog.csdn.net/william001zs/article/details/6213485 矢量 如果一条线段的端点是有次序之分的话,那么这种线段就称为 有向线段,如果 ...

随机推荐

  1. python3中的nonlocal 与 global

    nonlocal 与 global nonlocal翻译是非本地,global翻译是全局,它们都是python3的新特性.如果以类C语言的思维去看这2个关键字,很可能觉得它们差不多.但实际上它们很不一 ...

  2. L20 梯度下降、随机梯度下降和小批量梯度下降

    airfoil4755 下载 链接:https://pan.baidu.com/s/1YEtNjJ0_G9eeH6A6vHXhnA 提取码:dwjq 梯度下降 (Boyd & Vandenbe ...

  3. 💕《给产品经理讲JVM》:垃圾收集器

    前言 在上篇中,我们把 JVM 中的垃圾收集算法有了一个大概的了解,又是一个阴雨连绵的周末,宅在家里的我们又开始了新一轮的学习: 产品大大:上周末我们说了垃圾收集算法,下面是不是要讲一下这些算法的应用 ...

  4. Jmeter发送jdbc请求进行大批量造数

    创建批量造数脚本,一个简单的结构如下图所示, 1.线程组(10个线程重复运行2次,相当于造20个数) 2.用户定义变量(这是全局变量,用于后面随机筛选用) 3.数据库连接配置 4.计数器(用于主键递增 ...

  5. C - N皇后问题 DFS

    在N*N的方格棋盘放置了N个皇后,使得它们不相互攻击(即任意2个皇后不允许处在同一排,同一列,也不允许处在与棋盘边框成45角的斜线上. 你的任务是,对于给定的N,求出有多少种合法的放置方法. Inpu ...

  6. JavaScript中的作用域和作用域链(边学边写)[看着别人的博客纯手敲]

    作用域是JavaScript最重要的概念之一,想要学好JavaScript就需要理解JavaScript作用域和作用域的工作原理.今天这篇文章对JavaScript作用域和作用域链简单的介绍,希望能帮 ...

  7. @SessionAttributes 的使用

    @SessionAttributes 注解只用作用在 类 上,作用是将指定的 Model 的键值对保存在 session 中.可以让其他请求共用 session 中的键值对. 指定保存的属性名 作用是 ...

  8. 【题解】P1972 [SDOI2009]HH的项链 - 树状数组

    P1972 [SDOI2009]HH的项链 声明:本博客所有题解都参照了网络资料或其他博客,仅为博主想加深理解而写,如有疑问欢迎与博主讨论✧。٩(ˊᗜˋ)و✧*。 题目描述 \(HH\) 有一串由各种 ...

  9. 简谈” Top K“

    Top K 快速选择和堆排序都可以求解 Kth Element 和 TopK Elements 问题. 题见215. Kth Largest Element in an Array (Medium) ...

  10. go 基础 结构体

    结构体是类型中带有成员的复合类型.go语言使用结构体和结构体成员来描述真实世界的实体和实体对应的各种属性. go语言中的类型可以被实例化,使用new和&构造类型实例的类型是类型的指针. 结构体 ...