# -*- coding: utf-8 -*-
# @Time : 2019-09-18 16:55
# @Author : Jayce Wong
# @ProjectName : job
# @FileName : segment_cross.py
# @Blog : https://blog.51cto.com/jayce1111
# @Github : https://github.com/SysuJayce
"""
Q:给定两个线段的坐标(也就是四个点的直角坐标系坐标),判断这两个线段是否相交
### 判断线段是否相交可以利用向量的叉乘 ###
假定输入为P1、P2、Q1、Q2四个点的坐标,P1P2为一条线段,Q1Q2为另一条线段
两条线段相交只有两种情况
1. 其中一条线段的某一端点在另一条线段上;
2. 两条线段形成X形。
首先判断这四个点是否在另一条线段上,也就是说,判断P1是否在线段Q1Q2上,P2是否在线段Q1Q2上...
如果上述判断为真,那么这两条线段相交。【解决了第一种情况】
如果没有点在另一条线段上,那么进行叉乘判断。
先固定线段Q1Q2,然后以Q1为轴,计算Q1P1和Q1Q2、Q1P2和Q1Q2的叉乘是否异号;
然后固定线段P1P2,然后以P1为轴,计算P1Q1和P1P2、P1Q2和P1P2的叉乘是否异号。
当上述的叉乘都异号的时候,两条线段相交。
【解决了第二种情况】
"""
class Point:
def __init__(self, x, y):
self.x = x
self.y = y
def __sub__(self, other):
return Point(self.x - other.x, self.y - other.y)
class Segment:
def __init__(self, point1, point2):
self.point1 = point1
self.point2 = point2
self.x = point1.x - point2.x
self.y = point1.y - point2.y
def crossProduct(v1, v2):
return v1.x * v2.y - v2.x * v1.y
def onSegment(p, seg):
"""
判断点在不在一条线段上,关键在于:
1. 三点是否共线
2. 点p是否在线段的延长线上。
只要满足了三点共线,且点p不在延长线上,那么点p就在线段上。
判断三点共线可以用向量的叉乘,三点共线即两个向量平行,也就是叉乘结果为零向量(对应到二维就是零)
当点p的横纵坐标都在线段端点之间的时候,点p不在延长线上。
:param p:
:param seg:
:return:
"""
# 先确保点p不在延长线上
if min(seg.point1.x, seg.point2.x) <= p.x <= max(seg.point1.x, seg.point2.x)\
and min(seg.point1.y, seg.point2.y) <= p.y <= max(seg.point1.y, seg.point2.y):
# 然后确保这三个点形成的向量两两平行,这里只要这三个向量中任意两个平行,第三个一定也平行
if crossProduct(p - seg.point1, p - seg.point2) == 0:
return True
else:
return False
else:
return False
def isCross(p1, p2, q1, q2):
p1p2 = Segment(p1, p2)
q1q2 = Segment(q1, q2)
p1q1 = Segment(p1, q1)
p1q2 = Segment(p1, q2)
q1p1 = Segment(q1, p1)
q1p2 = Segment(q1, p2)
# 判断是否存在端点位于另一条线段上,是的话则两条线段相交
if any([onSegment(p1, q1q2), onSegment(p2, q1q2),
onSegment(q1, p1p2), onSegment(q2, p1p2)]):
return True
# 否则固定线段P1P2,判断Q1和Q2是否在P1P2的两侧(计算叉乘)
# 然后固定线段Q1Q2,判断P1和P2是否在Q1Q2的两侧
# 如果上面的判断均为真,那么这两条线段形成一个X
return (crossProduct(p1p2, p1q1) * crossProduct(p1p2, p1q2) < 0)\
and (crossProduct(q1q2, q1p1) * crossProduct(q1q2, q1p2) < 0)
def main():
p1 = Point(0, 0)
p2 = Point(2, 2)
q1 = Point(1, 1)
q2 = Point(0, 2)
if isCross(p1, p2, q1, q2):
print('Yes')
else:
print('No')
if __name__ == '__main__':
main()
标题名称:判断线段是否相交
浏览地址:
http://cdweb.net/article/iipiso.html