-
Notifications
You must be signed in to change notification settings - Fork 15
/
halfplaneintersect.py
171 lines (142 loc) · 6.3 KB
/
halfplaneintersect.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
# The MIT License (MIT)
#
# Copyright (c) 2013 Mak Nazecic-Andrlon
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to
# deal in the Software without restriction, including without limitation the
# rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
# sell copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
# IN THE SOFTWARE.
"""This module solves 2D linear programming using half-plane intersection."""
from __future__ import division
import itertools
from numpy import dot, clip, array, sqrt
from numpy.linalg import det
class InfeasibleError(RuntimeError):
"""Raised if an LP problem has no solution."""
pass
class Line(object):
"""A line in space."""
def __init__(self, point, direction):
super(Line, self).__init__()
self.point = array(point)
self.direction = normalized(array(direction))
def __repr__(self):
return "Line(%s, %s)" % (self.point, self.direction)
def halfplane_optimize(lines, optimal_point):
"""Find the point closest to optimal_point in the intersection of the
closed half-planes defined by lines which are in Hessian normal form
(point-normal form)."""
# We implement the quadratic time (though linear expected given randomly
# permuted input) incremental half-plane intersection algorithm as laid
# out in http://www.mpi-inf.mpg.de/~kavitha/lecture3.ps
point = optimal_point
for i, line in enumerate(lines):
# If this half-plane already contains the current point, all is well.
if dot(point - line.point, line.direction) >= 0:
# assert False, point
continue
# Otherwise, the new optimum must lie on the newly added line. Compute
# the feasible interval of the intersection of all the lines added so
# far with the current one.
prev_lines = itertools.islice(lines, i)
left_dist, right_dist = line_halfplane_intersect(line, prev_lines)
# Now project the optimal point onto the line segment defined by the
# the above bounds. This gives us our new best point.
point = point_line_project(line, optimal_point, left_dist, right_dist)
return point
def point_line_project(line, point, left_bound, right_bound):
"""Project point onto the line segment defined by line, which is in
point-normal form, and the left and right bounds with respect to line's
anchor point."""
# print("left_bound=%s, right_bound=%s" % (left_bound, right_bound))
new_dir = perp(line.direction)
# print("new_dir=%s" % new_dir)
proj_len = dot(point - line.point, new_dir)
# print("proj_len=%s" % proj_len)
clamped_len = clip(proj_len, left_bound, right_bound)
# print("clamped_len=%s" % clamped_len)
return line.point + new_dir * clamped_len
def line_halfplane_intersect(line, other_lines):
"""Compute the signed offsets of the interval on the edge of the
half-plane defined by line that is included in the half-planes defined by
other_lines.
The offsets are relative to line's anchor point, in units of line's
direction.
"""
# We use the line intersection algorithm presented in
# http://stackoverflow.com/a/565282/126977 to determine the intersection
# point. "Left" is the negative of the canonical direction of the line.
# "Right" is positive.
left_dist = float("-inf")
right_dist = float("inf")
for prev_line in other_lines:
num1 = dot(prev_line.direction, line.point - prev_line.point)
den1 = det((line.direction, prev_line.direction))
# num2 = det((perp(prev_line.direction), line.point - prev_line.point))
# den2 = det((perp(line.direction), perp(prev_line.direction)))
# assert abs(den1 - den2) < 1e-6, (den1, den2)
# assert abs(num1 - num2) < 1e-6, (num1, num2)
num = num1
den = den1
# Check for zero denominator, since ZeroDivisionError (or rather
# FloatingPointError) won't necessarily be raised if using numpy.
if den == 0:
# The half-planes are parallel.
if num < 0:
# The intersection of the half-planes is empty; there is no
# solution.
raise InfeasibleError
else:
# The *half-planes* intersect, but their lines don't cross, so
# ignore.
continue
# Signed offset of the point of intersection, relative to the line's
# anchor point, in units of the line's direction.
offset = num / den
if den > 0:
# Point of intersection is to the right.
right_dist = min((right_dist, offset))
else:
# Point of intersection is to the left.
left_dist = max((left_dist, offset))
if left_dist > right_dist:
# The interval is inconsistent, so the feasible region is empty.
raise InfeasibleError
return left_dist, right_dist
def perp(a):
return array((a[1], -a[0]))
def norm_sq(x):
return dot(x, x)
def norm(x):
return sqrt(norm_sq(x))
def normalized(x):
l = norm_sq(x)
assert l > 0, (x, l)
return x / sqrt(l)
if __name__ == '__main__':
lines = [
Line((-2, 0), (-1, 1)),
Line((0, -1), (1, 0))
]
point = array((1, 0))
result = halfplane_optimize(lines, point)
print(result, norm(result))
# a = point_line_project(lines[0], point, -20, 20)
# print((a - lines[0].point)/(-perp(lines[0].direction)))
# print(a)
a = point_line_project(lines[1], point, -10000, -3)
# print((a - lines[1].point)/(-perp(lines[1].direction)))
print(a)