加入多边形时Boost几何缓冲区的行为不符合预期

问题描述 投票:0回答:1

我有一组多边形,我想找到它们的并集。为了允许相互接触的多边形存在一些浮点误差,我使用

boost::geometry::buffer
以及通过
boost::geometry::strategy::buffer::distance_symmetric<double>
的通货膨胀策略。对于某些输入,缓冲区操作会产生我不理解的结果: enter image description here

有 7 个多边形需要连接。并集(通过

boost::geometry::union_
)的行为符合预期,但在缓冲连接中,最后两个多边形似乎丢失了。

有趣的是,如果我稍微扰动 Polygon 4(x 和 y 移动 +0.01),缓冲区的行为就会符合预期。 enter image description here

或者将其从集合中完全删除: enter image description here

多边形设置和缓冲区操作的代码片段:

  using Point = boost::geometry::model::d2::point_xy<double>;
  using Polygon = boost::geometry::model::polygon<Point>;
  double epsilon = 0.3;

  boost::geometry::strategy::buffer::join_miter join_strategy;
  boost::geometry::strategy::buffer::distance_symmetric<double>
      inflate_strategy(epsilon);
  boost::geometry::strategy::buffer::end_flat end_strategy;
  boost::geometry::strategy::buffer::side_straight side_strategy;
  boost::geometry::strategy::buffer::point_circle point_strategy;

  boost::geometry::model::multi_polygon<Polygon> source_set, joined_inflated;

  source_set.push_back(
      Polygon{{Point(12.50, 36.50), Point(12.50, 30.50), Point(6.50, 30.50),
               Point(6.50, 36.50), Point(12.50, 36.50)}});
  source_set.push_back(
      Polygon{{Point(6.50, 27.00), Point(6.50, 33.50), Point(12.50, 33.50),
               Point(12.50, 27.00), Point(6.50, 27.00)}});
  source_set.push_back(
      Polygon{{Point(12.50, 30.00), Point(12.50, 24.00), Point(6.50, 24.00),
               Point(6.50, 30.00), Point(12.50, 30.00)}});
  source_set.push_back(
      Polygon{{Point(24.00, 24.00), Point(9.50, 24.00), Point(9.50, 30.00),
               Point(24.00, 30.00), Point(24.00, 24.00)}});
  source_set.push_back(
      Polygon{{Point(27.0, 30.0), Point(27.0, 24.0), Point(21.0, 24.0),
               Point(21.0, 30.0), Point(27.0, 30.0)}});
  source_set.push_back(
      Polygon{{Point(28.10, 30.73), Point(26.90, 26.23), Point(21.10, 27.77),
               Point(22.30, 32.27), Point(28.10, 30.73)}});
  source_set.push_back(
      Polygon{{Point(28.20, 34.50), Point(28.20, 28.50), Point(22.20, 28.50),
               Point(22.20, 34.50), Point(28.20, 34.50)}});   

  boost::geometry::buffer(source_set, joined_inflated, inflate_strategy,
                          side_strategy, join_strategy, end_strategy,
                          point_strategy);

为什么缓冲连接会这样?

c++ boost boost-geometry
1个回答
0
投票

根据您的 Godbolt 链接,1.78 修复了它:1.77 与 1.78 https://godbolt.org/z/9ofqx75z1

更改历史记录有一些潜在相关的项目:

错误

  • 906 使用 clang-12 在 macos-11 上的缓冲区结果无效。
  • 921 c++20(各种编译器)和 gcc-5 的编译错误。
  • 集合操作和缓冲区中的各种修复。

我认为它属于“集合操作和缓冲区中的各种修复”,然后:

git log --oneline --graph --left-right boost-1.77.0..boost-1.78.0 -- $(git ls-files | grep buffer)
> 2a7db45d0 [test] take car alternate tests are test properly in old (rescaling) and new (no rescaling) regime
> ef1b8e33f [side] make the default for no-rescaling triangle to avoid regressions when rescaling is turned off
> 915564a02 [test][buffer] Add test case that was failing on macos-11, clang-12.0.5, x86-64.
> cc21b05ab [buffer] Increase traversable turns distance threshold.
> 716c79136 [test] enhance/fix robustness tests
> e99cfde12 [intersection] use balance between distance-to-end and length-of-segments to determine to use a or b
> 03d6e82f2 [coordinate] deprecate util/promote_floating_point.hpp
> 5110ec7da Add missing headers to satisfy Boost header policy.
> 08f7e66f7 [test] Drop library dependencies in tests.

手动平分

显示 03d6e82f2 是最后一个损坏的版本。由于 03d6e82f2..e99cfde12 之间没有其他提交,这意味着修复是实际上

intersection

commit e99cfde120ba347800a1b2472cffed93729e3ad1
Author: Barend Gehrels <[email protected]>
Date:   Wed Sep 8 11:54:19 2021 +0200

    [intersection] use balance between distance-to-end and length-of-segments to determine to use a or b

代码有一些注释,例如

// When calculating the intersection, the information of "a" or "b" can be used.
// Theoretically this gives equal results, but due to floating point precision
// there might be tiny differences. These are edge cases.

但真正的解释出现在相关 PR 中:https://github.com/boostorg/geometry/pull/897。简而言之,这是 Mysql 开发人员发现的稳健性修复。

© www.soinside.com 2019 - 2024. All rights reserved.