2018-12-11 15:57:58 +08:00
|
|
|
''' mbinary
|
|
|
|
#########################################################################
|
|
|
|
# File : min_distance_of_n_points.py
|
|
|
|
# Author: mbinary
|
|
|
|
# Mail: zhuheqin1@gmail.com
|
2019-01-31 12:09:46 +08:00
|
|
|
# Blog: https://mbinary.xyz
|
2018-12-11 15:57:58 +08:00
|
|
|
# Github: https://github.com/mbinary
|
|
|
|
# Created Time: 2018-11-24 22:03
|
|
|
|
# Description:
|
|
|
|
#########################################################################
|
|
|
|
'''
|
2018-11-07 16:52:55 +08:00
|
|
|
from random import randint
|
|
|
|
from time import time
|
|
|
|
from functools import total_ordering
|
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
@total_ordering
|
|
|
|
class point:
|
2020-04-15 12:28:20 +08:00
|
|
|
def __init__(self, x, y):
|
|
|
|
self.x = x
|
|
|
|
self.y = y
|
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def __neg__(self):
|
|
|
|
return pont(-self.x, -self.y)
|
2020-04-15 12:28:20 +08:00
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def __len__(self):
|
|
|
|
return self.norm(2)
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def __lt__(self, p):
|
|
|
|
return self.x < p.x or (self.x == p.x and self.y < p.y)
|
|
|
|
|
|
|
|
def __eq__(self, p):
|
|
|
|
return self.x == p.x and self.y == p.y
|
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def __hash__(self):
|
2020-04-15 12:28:20 +08:00
|
|
|
return hash((self.x, self.y))
|
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def __repr__(self):
|
2020-04-15 12:28:20 +08:00
|
|
|
return 'point({},{})'.format(self.x, self.y)
|
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def __str__(self):
|
|
|
|
return self.__repr__()
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def norm(self, n=2):
|
|
|
|
if n <= 0:
|
|
|
|
return max(abs(self.x), abs(self.y))
|
2018-11-07 16:52:55 +08:00
|
|
|
return (abs(self.x)**n+abs(self.y)**n)**(1/n)
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def distance(self, p):
|
2018-11-07 16:52:55 +08:00
|
|
|
return ((self.x-p.x)**2+(self.y-p.y)**2)**0.5
|
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def minDistance_n2(points):
|
|
|
|
n = len(points)
|
2020-04-15 12:28:20 +08:00
|
|
|
if n <= 1:
|
|
|
|
return 0
|
|
|
|
p, q = points[:2]
|
2018-11-07 16:52:55 +08:00
|
|
|
minD = points[0].distance(points[1])
|
|
|
|
for i in range(n-1):
|
2020-04-15 12:28:20 +08:00
|
|
|
for j in range(i+1, n):
|
2018-11-07 16:52:55 +08:00
|
|
|
d = points[i].distance(points[j])
|
2020-04-15 12:28:20 +08:00
|
|
|
if d < minD:
|
2018-11-07 16:52:55 +08:00
|
|
|
minD = d
|
|
|
|
p = points[i]
|
2020-04-15 12:28:20 +08:00
|
|
|
q = points[j]
|
|
|
|
return minD, p, q
|
2018-11-07 16:52:55 +08:00
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def findif(points, f, reverse=False):
|
2018-11-07 16:52:55 +08:00
|
|
|
n = len(points)
|
2020-04-15 12:28:20 +08:00
|
|
|
rg = range(n-1, -1, -1) if reverse else range(n)
|
|
|
|
for i in rg:
|
|
|
|
if not f(points[i]):
|
|
|
|
return points[i+1:] if reverse else points[:i]
|
|
|
|
return points.copy() # note that don't return exactly points, return a copy one
|
|
|
|
|
|
|
|
|
|
|
|
def floatEql(f1, f2, epsilon=1e-6):
|
|
|
|
return abs(f1-f2) < epsilon
|
2018-11-07 16:52:55 +08:00
|
|
|
|
|
|
|
|
|
|
|
def minDistance_nlogn(n_points):
|
|
|
|
def _min(pts):
|
|
|
|
n = len(pts)
|
2020-04-15 12:28:20 +08:00
|
|
|
if n == 2:
|
|
|
|
return pts[0].distance(pts[1]), pts[0], pts[1]
|
|
|
|
if n == 3:
|
2018-11-07 16:52:55 +08:00
|
|
|
minD = pts[0].distance(pts[1])
|
2020-04-15 12:28:20 +08:00
|
|
|
p, q = pts[0], pts[1]
|
2018-11-07 16:52:55 +08:00
|
|
|
d2 = pts[2].distance(pts[1])
|
2020-04-15 12:28:20 +08:00
|
|
|
if minD > d2:
|
2018-11-07 16:52:55 +08:00
|
|
|
minD = d2
|
2020-04-15 12:28:20 +08:00
|
|
|
p, q = pts[1], pts[2]
|
2018-11-07 16:52:55 +08:00
|
|
|
d2 = pts[0].distance(pts[2])
|
2020-04-15 12:28:20 +08:00
|
|
|
if minD > d2:
|
|
|
|
return d2, pts[0], pts[2]
|
|
|
|
else:
|
|
|
|
return minD, p, q
|
2018-11-07 16:52:55 +08:00
|
|
|
n2 = n//2
|
2020-04-15 12:28:20 +08:00
|
|
|
mid = (pts[n2].x + pts[n2-1].x)/2
|
2018-11-07 16:52:55 +08:00
|
|
|
s1 = pts[:n2]
|
|
|
|
s2 = pts[n2:]
|
2020-04-15 12:28:20 +08:00
|
|
|
minD, p, q = _min(s1)
|
2018-11-07 16:52:55 +08:00
|
|
|
d2, p2, q2 = _min(s2)
|
2020-04-15 12:28:20 +08:00
|
|
|
# print('\n\n',minD,p,q,s1)
|
|
|
|
# print(d2,p2,q2,s2)
|
|
|
|
if minD > d2:
|
|
|
|
minD, p, q = d2, p2, q2
|
2018-11-07 16:52:55 +08:00
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
linePoints = findif(s1, lambda pt: floatEql(pt.x, mid), reverse=True)
|
|
|
|
linePoints += findif(s2, lambda pt: floatEql(pt.x, mid))
|
2018-11-07 16:52:55 +08:00
|
|
|
n = len(linePoints)
|
2020-04-15 12:28:20 +08:00
|
|
|
if n > 1:
|
|
|
|
for i in range(1, n):
|
|
|
|
dis = linePoints[i].y - linePoints[i-1].y
|
|
|
|
if dis < minD:
|
2018-11-07 16:52:55 +08:00
|
|
|
minD = dis
|
2020-04-15 12:28:20 +08:00
|
|
|
p, q = linePoints[i-1], linePoints[i]
|
|
|
|
leftPoints = findif(s1, lambda pt: pt.x >= mid-minD, reverse=True)
|
|
|
|
rightPoints = findif(s2, lambda pt: pt.x <= mid+minD)
|
2018-11-07 16:52:55 +08:00
|
|
|
for lp in leftPoints:
|
2020-04-15 12:28:20 +08:00
|
|
|
y1, y2 = lp.y-minD, lp.y+minD
|
2018-11-07 16:52:55 +08:00
|
|
|
for rp in rightPoints:
|
2020-04-15 12:28:20 +08:00
|
|
|
if y1 < rp.y < y2:
|
2018-11-07 16:52:55 +08:00
|
|
|
dis = lp.distance(rp)
|
2020-04-15 12:28:20 +08:00
|
|
|
if dis < minD:
|
2018-11-07 16:52:55 +08:00
|
|
|
minD = dis
|
2020-04-15 12:28:20 +08:00
|
|
|
p, q = lp, rp
|
|
|
|
return minD, p, q
|
2018-11-07 16:52:55 +08:00
|
|
|
return _min(sorted(n_points))
|
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def test(f=minDistance_n2):
|
|
|
|
print('\ntest : ', f.__name__)
|
|
|
|
begin = time()
|
|
|
|
minD, p, q = f(points)
|
|
|
|
print('time : {:.6f} s'.format(time()-begin))
|
2020-04-15 12:28:20 +08:00
|
|
|
print('result: {:.2f} {} {}\n'.format(minD, p, q))
|
2018-11-07 16:52:55 +08:00
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def genData(n, unique=True):
|
2018-12-11 15:28:05 +08:00
|
|
|
upper = 1000000
|
2018-11-07 16:52:55 +08:00
|
|
|
if unique:
|
|
|
|
points = set()
|
|
|
|
for i in range(n):
|
2020-04-15 12:28:20 +08:00
|
|
|
points.add(point(randint(1, upper), randint(1, upper)))
|
2018-11-07 16:52:55 +08:00
|
|
|
return list(points)
|
2020-04-15 12:28:20 +08:00
|
|
|
else:
|
|
|
|
return [point(randint(1, upper), randint(1, upper)) for i in range(n)]
|
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
if __name__ == '__main__':
|
2018-12-11 15:28:05 +08:00
|
|
|
n = 1000
|
2018-11-07 16:52:55 +08:00
|
|
|
points = genData(n, unique=True)
|
|
|
|
print('min distance of {} points'.format(n))
|
2020-04-15 12:28:20 +08:00
|
|
|
# print(sorted(points))
|
2018-11-07 16:52:55 +08:00
|
|
|
test(minDistance_n2)
|
|
|
|
test(minDistance_nlogn)
|