Skip to content

Latest commit

 

History

History
49 lines (43 loc) · 3.9 KB

README.md

File metadata and controls

49 lines (43 loc) · 3.9 KB

简介

该仓库包含了一个性能较好的求交算法,用于计算大批量的几何图形之间的相交情况。它的优势在于一次性计算大量图形两两之间的相交情况,相同测试数据下比boost.geometry.spatial_indexes.rtree快很多。

组成

  1. 算法核心在intersect/bound/RangeBound.cpp中。
  2. 测试代码在intersect/test中,包含准确性测试、性能测试,测试结果会打印在TestResult.log中,如:

对84834个实体求交 -> 暴力遍历 总计:37997ms
对84834个实体求交 -> 外包排斥 总计:11491ms
对84834个实体求交 -> Range2d 总计:200ms
对84834个实体求交 -> boost.geometry.index.rtree-Packing 总计:589ms

  1. 为了支持测试,项目中把84834个图形序列化到数据库TestData.db3中,intersect/sqlite/SindySQLite.cpp提供相应的读写函数。
  2. intersect/common包含了一些计时等工具函数。

算法思路

  • 在计算几何中,每个图形都有一个外接矩形:Bounding,图形是否相交可以先判断Bounding范围内是否相交。
  • Bounding相交可以用快速排斥的思想求出,请注意下面的outExtents函数,最佳情况下只需要一次浮点数比较就能判断Bounding是否相交。
// SindyExtents.cpp
bool Extents::outExtents(const Extents& ext, double tol) const
{
    if (ext.m_max.x < m_min.x - tol || ext.m_min.x > m_max.x + tol)
        return true;
    if (ext.m_max.y < m_min.y - tol || ext.m_min.y > m_max.y + tol)
        return true;
    if (ext.m_max.z < m_min.z - tol || ext.m_min.z > m_max.z + tol)
        return true;
    return false;
}
  1. 把扫描线求交的思想和包围盒排序相结合,是本算法的核心思想。
  2. 用户通过sindy::Range2d使用该算法,把每个Bounding的最小点、最大点的X值分别创建节点对象放进std::vector中,先对std::vector中的元素按X值从小到大排序,再创建Y方向的临时容器std::multimap,而后从小到大遍历所有Bounding的X值。
  3. 当遍历到某个item的最小点时,与此图形相交的其它图形开始创建两两关联,请注意mapY2Item.lower_bound(pSrcItem->m_dMinY),此时不会找到min.x大于item.min.x的目标图形,但是后续遍历到目标图形时仍然会正确地创建两者的关联。
  4. 当遍历到item的最大点时,与此图形相交的其它图形都找完了,所以把所有Y值为item.max.y的图形从mapY2Item中移除。
  5. 请注意,这里是关键点:遍历到最小点时,会将item的最大点的Y值放进临时容器(请思考为什么是max.y);遍历到最大点时,会将当前的max.y从临时容器移除。
  6. 核心实现都在Range2d::getIntersectItem(创建节点数据时请使用配套的setItem函数),其它函数都是各种变体。

使用

  1. 为了支持扩展,使用者需要把每个待计算的图形或图形数据类继承自IBoundItem,并至少重写getExtents函数,该函数用于获取图形的Bounding,这是必须的。getId可以返回某种标记,可以是数组索引、图形ID等,当你需要一个额外的值时。
  2. 具体用法请参考测试代码。

构建

  1. 本项目采用CMake构建,CMake支持跨平台构建,几乎所有主流IDE都支持它,且不受IDE本身的版本影响。
  2. 为了更高效地完成编码,项目中使用了C++11的语法(特别是在测试用例中),如果您目前还在使用C++98标准,需要根据编译器提示做一些简单的等价替换。
  3. 为了对比测试性能,项目中使用了boost.geometry库,但并未开启该部分测试代码。如要开启,请在CMakeLists中提供编译好的头文件、库路径。

其它

  • 对算法的错误指正、改良、技术支持或其它疑问,欢迎咨询QQ群:571208653。
  • 如果觉得不错,请点Star让更多人看到。