【问题标题】:Computing a polygon that surrounds a multi-point line计算围绕多点线的多边形
【发布时间】:2011-08-11 22:18:59
【问题描述】:

我正在尝试计算围绕连接多个点的线的多边形(例如 GPX 轨道)。 下图显示了一个示例,其中轨道为红线,所需的多边形为蓝色。

为简化起见,红点由 x 和 y 表示 - 而不是由纬度/经度表示。

如果我只有指定路径的三个点的列表,我该如何计算这样的环境(浅蓝色多边形)?

为此类计算提供算法的部分解决方案(例如,仅针对两点)或有关数学库(Java 中)的提示也会使我向前迈出一步。

进一步的假设:

  • 轨道没有交叉路口。

更新: 使用 Rogach 和 xan 提出的方法,如果线之间的角度小于 90 度或大于 270 度,我会遇到一些问题: 如您所见,多边形与自身相交,这导致了一个严重的问题。

在我看来,使用java.awt.geom.Area 是更好的方法:

我的解决方案(基于 Rogach 的代码):

对于连接轨道两点的每条线,我计算一个周围的多边形。之后,我将计算出的多边形添加(区域联合)到一个区域,该区域为我完成所有必要的计算。由于该区域在添加新多边形时严格使用“或”算法,因此我不必关心上面更新中介绍的多边形的“自相交”。

Area area = new Area();
for (int i = 1; i < points.size(); i++) {
    Point2D point1 = points.get(i - 1);
    Point2D point2 = points.get(i);

    Line2D.Double ln = new Line2D.Double(point1.getX(), point1.getY(), point2.getX(), point2.getY());
    double indent = 15.0; // distance from central line
    double length = ln.getP1().distance(ln.getP2());

    double dx_li = (ln.getX2() - ln.getX1()) / length * indent;
    double dy_li = (ln.getY2() - ln.getY1()) / length * indent;

    // moved p1 point
    double p1X = ln.getX1() - dx_li;
    double p1Y = ln.getY1() - dy_li;

    // line moved to the left
    double lX1 = ln.getX1() - dy_li;
    double lY1 = ln.getY1() + dx_li;
    double lX2 = ln.getX2() - dy_li;
    double lY2 = ln.getY2() + dx_li;

    // moved p2 point
    double p2X = ln.getX2() + dx_li;
    double p2Y = ln.getY2() + dy_li;

    // line moved to the right
    double rX1_ = ln.getX1() + dy_li;
    double rY1 = ln.getY1() - dx_li;
    double rX2 = ln.getX2() + dy_li;
    double rY2 = ln.getY2() - dx_li;

    Path2D p = new Path2D.Double();
    p.moveTo(lX1, lY1);
    p.lineTo(lX2, lY2);
    p.lineTo(p2X, p2Y);
    p.lineTo(rX2, rY2);
    p.lineTo(rX1_, rY1);
    p.lineTo(p1X, p1Y);
    p.lineTo(lX1, lY1);

    area.add(new Area(p));
}

【问题讨论】:

  • 这些线会相交吗?或者你不在乎多边形是否重叠?
  • 最后一个没有交点的多边形就是目标。
  • 如果路径的两个部分相交但在路径中没有相互连接,您希望它是什么样子?
  • 一般我假设路径(红线)是无交叉的。

标签: java math polygon


【解决方案1】:

如我所见,这个问题类似于多边形缓冲问题。

我认为以下方法可以帮助您:

  • 对于轨道的每一段,找到两条线 - 一条在左边,一条在右边。
  • 然后,遍历您的偏移线,并解决交叉点。例如: http://img25.imageshack.us/img25/7660/temprhk.png
  • 在末端添加大写字母,您就完成了! :)

还有一些代码:

向左移动一行:

Line2D l; 
double indent; // distance from central line
double dx = ln.getX2() - ln.getX1();
double dy = ln.getY2() - ln.getY1();
double length = ln.getP1().distance(ln.getP2());
double newX1 = l.getX1() - indent*(dy/length);
double newY1 = l.getY1() + indent*(dx/length);
double newX2 = l.getX2() - indent*(dy/length);
double newY2 = l.getY2() + indent*(dx/length);
Line2D leftLine = new Line2D.Double(newX1, newY1, newX2, newY2);

要将其向右移动,请将最后 4 行代码中的“+”更改为“-”,反之亦然。

关于使用交点 - 如果两条线段相交,您只需输出交点。如果他们不这样做,那么情况就有点复杂了——当然,你仍然可以输出交叉点,但是在快速转弯的情况下,会出现奇怪的爆发。我在类似的情况下插入了一个圆弧段,但是代码又大又分散,所以我不能在这里粘贴。
或者,您可以按照图片上显示的方式进行操作 - 只需连接端点即可。


而且,顺便说一句,如果速度不是大问题,你可以使用更好的方法 - 对于每一行轨道,找到左右行,添加大写,将它们全部打包到Path2D,然后创建Area from路径 2D。
在这种情况下,您可以将这条“带帽子的线”作为三个区域的交点:矩形,其点恰好是左右线的端点,两个圆心在原始轨道段的末端。

当您计算所有线的面积时,只需使用 Area add() 方法将它们相交。

这种方法可以处理任何情况,甚至是轨道中的自相交和中断。

【讨论】:

  • Path2D/Area 方法效果更好(参见我的解决方案),这是我认为的解决方案。
【解决方案2】:

请参阅my answer 的类似问题“如何在任何线周围绘制轮廓”。

与 Rogach 在这里提供的想法相同,但也许不同的图纸和解释将有助于澄清它。

【讨论】:

    【解决方案3】:

    如果您不想编写 Rogach 所描述的缓冲代码,JTS 可以为您创造奇迹。 请参阅developer guide 了解快速介绍。

    【讨论】:

      【解决方案4】:

      半生不熟的建议:计算每个段的法线。然后,对于每个顶点V_i,插入其相邻段的法线以获得n_i(再次对其进行标准化)并在V_i +/- a*n_i处添加两个顶点,其中a是一些缩放因子。

      如果你加入这些点,你将不会得到完全你的蓝色多边形,但它可能已经足够了。

      您可能必须跟踪新顶点位于哪个“边”上。如果您可以在没有自相交的情况下闭合曲线,这将成为每个顶点的 point in polygon test

      【讨论】:

        猜你喜欢
        • 2014-02-08
        • 2020-02-04
        • 2011-08-12
        • 2019-08-30
        • 2018-01-17
        • 2013-11-10
        • 1970-01-01
        • 1970-01-01
        • 2011-10-30
        相关资源
        最近更新 更多