我们在初等几何中学过如何判断两条直线是否相交。在欧几里得平面上,两条直线要么平行,要么相交,要么重合。这是欧几里得第五公设的推论。相交的两条直线恰好有一个交点,而重合的两条直线有无数个交点。判断两条直线位置关系的代数方法是:联立两条之直线方程,无解,则两条直线平行,存在唯一解则两条直线相交,存在两个及以上的解,则两直线重合。
分为矩形判定和叉积判定两步
判断分别以两线段为对角线的两个矩形是否相交,若不相交,则两线段一定不相交。两个矩形是否相交的条件是:任一矩形的最右端都大于另一矩形的最左端,且任一矩形最高端大于另一矩形的最低端;只要其中任一条件不满足,则两矩形不相交,也即两线段不相交。
这一步判定不相交的两条线段直接返回False
。
经过上面判定未得到结果的两条线段那进入此步进行判定。
这一步的目的是确定一条线段的两个端点是否在另一线段的两侧,如果两条线段的端点互相位于另一条线段的两侧,则这两条线段相交。
def cross(p1,p2,p3): # 叉积判定
x1=p2[0]-p1[0]
y1=p2[1]-p1[1]
x2=p3[0]-p1[0]
y2=p3[1]-p1[1]
return x1*y2-x2*y1
def segment(p1,p2,p3,p4): #判断两线段是否相交
#矩形判定,以l1、l2为对角线的矩形必相交,否则两线段不相交
if(max(p1[0],p2[0])>=min(p3[0],p4[0]) #矩形1最右端大于矩形2最左端
and max(p3[0],p4[0])>=min(p1[0],p2[0]) #矩形2最右端大于矩形1最左端
and max(p1[1],p2[1])>=min(p3[1],p4[1]) #矩形1最高端大于矩形2最低端
and max(p3[1],p4[1])>=min(p1[1],p2[1])): #矩形2最高端大于矩形1最低端
if(cross(p1,p2,p3)*self.cross(p1,p2,p4)<=0
and cross(p3,p4,p1)*self.cross(p3,p4,p2)<=0):
D=1
else:
D=0
else:
D=0
return D
矩形的特殊之处在于其有2
条对角线,小于其边数4
。对于凸多边形,只要一条线段穿越矩形必定与某一条对角线相交。因此我们将线段与矩形相交问题转化为线段与矩形对角线的交点问题,为了排除特殊情况,在检测线段与对角线交点前,我们检测线段的两个端点是否在矩形内。
def check(l1,l2,sq):
# step 1 check if end point is in the square
if ( l1[0] >= sq[0] and l1[1] >= sq[1] and l1[0] <= sq[2] and l1[1] <= sq[3]) or
( l2[0] >= s1[0] and l2[1] >= s1[1] and l2[0] <= sq[2] and l2[1] <= sq[3]):
return 1
else:
# step 2 check if diagonal cross the segment
p1 = [sq[0],sq[1]]
p2 = [sq[2],sq[3]]
p3 = [sq[2],sq[1]]
p4 = [sq[0],sq[3]]
if segment(l1,l2,p1,p2) or segment(l1,l2,p3,p4):
return 1
else:
return 0
其中 sq=[xleftdown,yleftdown,xrightup,yrightup] s q = [ x l e f t d o w n , y l e f t d o w n , x r i g h t u p , y r i g h t u p ] ,代表矩形左下角和右上角的坐标。