SymPy中多边形的分段与内部的交点

时间:2018-01-15 22:26:20

标签: python geometry sympy computational-geometry

我想找到多边形覆盖的线段的长度。 为此,我使用以下的sympy代码。不幸的是,它认为多边形是一组线而不是一个区域,所以我只得到了交点。

>>> import sympy
>>> t = sympy.geometry.Polygon((0,0), (0,1), (1,0))
>>> l = sympy.geometry.Segment((1/3,1/3), (2,2))
>>> t.intersect(l)
{Point2D(1/2, 1/2)}

我试图使用多边形的closure比例,遗憾的是,这似乎没有实现。

>>> t.closure
NotImplementedError                       Traceback (most recent call last)

我或许可以尝试使用sympy返回的点来计算结果段,但是,似乎有很多极端情况,我很想让图书馆为我做这个。

我忽略了什么吗?

1 个答案:

答案 0 :(得分:1)

Polygon类可以称为PolygonalCurve;它目前不是要描述填充的平面形状。

  

多边形被视为封闭路径而不是2D区域

对于凸多边形,以下方法有效:

def intersection_length(l, t):
    endpoints = [p for p in [l.p1, l.p2] if t.encloses(p)]
    endpoints.extend(iter(t.intersect(l)))
    if len(endpoints) < 2:
        return 0
    else: 
        intersection = sympy.geometry.Segment(*endpoints)
    return intersection.length

它会在您的示例中返回sqrt(2)/6。这个想法是,如果交叉点是一个非平凡的线段,它的端点将来自(a)l与t的边界的交点; (b)由t包围的l的端点。因此,所有候选人都被收集,如果我们有两个,那么找到他们之间的距离。

(尽管可以在不形成Segment对象的情况下计算距离,但我认为无论如何都要很好)。

对于非凸多边形,我们可以有更复杂的交叉点,有几个部分。我也为这个案例编写了一个解决方案,但是你需要GitHub version of SymPy,因为它使用的是最近才添加的parameter_value方法(它将在SymPy 1.2中)。

这里的方法是像以前一样找到所有端点(可能有2个以上),根据线段的方向对它们进行排序,然后测试每个端点是否应该包含(通过考虑是否中点)这件作品是封闭的)。

import sympy

def intersection_length(segment, polygon):
    endpoints = [p for p in [segment.p1, segment.p2] if polygon.encloses(p)]
    endpoints.extend(iter(polygon.intersect(segment)))
    u = sympy.Symbol('u')
    endpoints.sort(key=lambda p: segment.parameter_value(p, u)[u])
    total_length = 0
    for a, b in zip(endpoints[:-1], endpoints[1:]):
        piece = sympy.geometry.Segment(a, b)
        if polygon.encloses(piece.midpoint):
            total_length += piece.length
    return total_length

t = sympy.geometry.Polygon((0, 0), (5, 0), (1, 1), (0, 5))
l = sympy.geometry.Segment((-1, 5), (5, -1))
print(intersection_length(l, t))