什么是四叉树?

四叉树可以有效解决这个问题。

四叉树每一层都把地图划分四块,根据地图尺寸来决定树的层数,层数越大划分越细。

但需要对某一范围的单位筛选时,只需要定位到与范围相交的树区域,再对其区域内的对象筛选即可。

四叉树的实现

 #pragma once

 #include "base.h"
 #include "math.h"

 template <class Value>
 class Tree4 {
 private:
     struct Pointer {
         Tree4 *LT, *RT, *LB, *RB;
         Pointer() :LT(nullptr), RT(nullptr), LB(nullptr), RB(nullptr)
         { }
         ~Pointer()
         {
             SAFE_DELETE(LT);
             SAFE_DELETE(RT);
             SAFE_DELETE(LB);
             SAFE_DELETE(RB);
         }
     };

 public:
     Tree4(): _rect(rect)
     {
         STD queue<Tree4 *> queue;
         queue.push(this);
         ; n != ; --n, c *= )
         {
             ; i != c; ++i)
             {
                 auto tree = queue.front();
                 tree->Root();
                 queue.pop();
                 queue.push(tree->_pointer.LT);
                 queue.push(tree->_pointer.RT);
                 queue.push(tree->_pointer.LB);
                 queue.push(tree->_pointer.RB);
             }
         }
     }

     template <class Range>
     bool Insert(const Value * value, const Range & range)
     {
         auto tree = Contain(range);
         auto ret = nullptr != tree;
         if (ret) { tree->_values.emplace_back(value); }
         return ret;
     }

     template <class Range>
     bool Remove(const Value * value, const Range & range)
     {
         auto tree = Contain(range);
         auto ret = nullptr != tree;
         if (ret) { ret = tree->Remove(value); }
         return ret;
     }

     template <class Range>
     bool Match(const Range & range, const STD function<bool(Value *)> & func)
     {
         if (!MATH intersect(_rect, range))
         {
             return true;
         }

         for (auto & value : _values)
         {
             if (!func(const_cast<Value *>(value)))
             {
                 return false;
             }
         }

         auto ret = true;
         if (!IsLeaf())
         {
             if (ret) ret = _pointer.LT->Match(range, func);
             if (ret) ret = _pointer.RT->Match(range, func);
             if (ret) ret = _pointer.LB->Match(range, func);
             if (ret) ret = _pointer.RB->Match(range, func);
         }
         return ret;
     }

     template <class Range>
     Tree4 * Contain(const Range & range)
     {
         Tree4<Value> * ret = nullptr;
         if (MATH contain(STD cref(_rect), range))
         {
             if (!IsLeaf())
             {
                 if (nullptr == ret) ret = _pointer.LT->Contain(range);
                 if (nullptr == ret) ret = _pointer.RT->Contain(range);
                 if (nullptr == ret) ret = _pointer.LB->Contain(range);
                 if (nullptr == ret) ret = _pointer.RB->Contain(range);
             }
             if (nullptr == ret)
                 ret = this;
         }
         return ret;
     }

 private:
     void Root()
     {
         _pointer.LT = new Tree4(MATH Rect(_rect.x, _rect.y, _rect.w * 0.5f, _rect.h * 0.5f));
         _pointer.LB = new Tree4(MATH Rect(_rect.x, _rect.y + _rect.h * 0.5f, _rect.w * 0.5f, _rect.h * 0.5f));
         _pointer.RT = new Tree4(MATH Rect(_rect.x + _rect.w * 0.5f, _rect.y, _rect.w * 0.5f, _rect.h * 0.5f));
         _pointer.RB = new Tree4(MATH Rect(_rect.x + _rect.w * 0.5f, _rect.y + _rect.h * 0.5f, _rect.w * 0.5f, _rect.h * 0.5f));
     }

     bool Remove(const Value * value)
     {
         auto iter = STD find(_values.begin(), _values.end(), value);
         auto ret = _values.end() != iter;
         if (ret) { _values.erase(iter); }
         return ret;
     }

     bool IsLeaf()
     {
         return nullptr == _pointer.LT
             || nullptr == _pointer.RT
             || nullptr == _pointer.LB
             || nullptr == _pointer.RB;
     }

     Tree4(const Tree4 &) = delete;
     Tree4(Tree4 &&) = delete;
     Tree4 &operator=(const Tree4 &) = delete;
     Tree4 &operator=(Tree4 &&) = delete;

 private:
     MATH Rect _rect;
     Pointer _pointer;
     STD list<const Value *> _values;
 };

代码简洁,通俗易懂,承让。

效果图

左侧全图遍历,右侧四叉树遍历,通过左上角的开销时间,差异很明显。

下载源码戳这里!!!

C++实现四叉树的更多相关文章

  1. 地图四叉树一般用在GIS中,在游戏寻路中2D游戏中一般用2维数组就够了

    地图四叉树一般用在GIS中,在游戏寻路中2D游戏中一般用2维数组就够了 四叉树对于区域查询,效率比较高. 原理图

  2. HTML5实现3D和2D可视化QuadTree四叉树碰撞检测

    QuadTree四叉树顾名思义就是树状的数据结构,其每个节点有四个孩子节点,可将二维平面递归分割子区域.QuadTree常用于空间数据库索引,3D的椎体可见区域裁剪,甚至图片分析处理,我们今天介绍的是 ...

  3. HT for Web可视化QuadTree四叉树碰撞检测

    QuadTree四叉树顾名思义就是树状的数据结构,其每个节点有四个孩子节点,可将二维平面递归分割子区域.QuadTree常用于空间数据库索引,3D的椎体可见区域裁剪,甚至图片分析处理,我们今天介绍的是 ...

  4. [转]一个四叉树Demo学习

    程序代码: http://www.codeproject.com/Articles/30535/A-Simple-QuadTree-Implementation-in-C 四叉树: using Sys ...

  5. [转]基于四叉树(QuadTree)的LOD地形实现

    实现基于四叉树的LOD地形时,我遇到的主要问题是如何修补地形裂缝. 本段我将描述使用LOD地形的优势,我实现LOD地形的思路,实现LOD地形核心模块的详细过程,以及修补地形裂缝的思路. 首先,LOD地 ...

  6. 线性四叉树十进制Morton码计算示例

    线性四叉树十进制Morton码计算,具体算法描述龚健雅<地理信息系统基础>P108

  7. 一个四叉树Demo学习

    程序代码: http://www.codeproject.com/Articles/30535/A-Simple-QuadTree-Implementation-in-C 四叉树: using Sys ...

  8. UVA 297 Quadtrees(四叉树建树、合并与遍历)

    <span style="font-size: 18pt; font-family: Arial, Helvetica, sans-serif; background-color: r ...

  9. [LeetCode] Quad Tree Intersection 四叉树相交

    A quadtree is a tree data in which each internal node has exactly four children: topLeft, topRight,  ...

随机推荐

  1. 制作 OpenStack Linux 镜像 - 每天5分钟玩转 OpenStack(151)

    这是 OpenStack 实施经验分享系列的第 1 篇. OpenStack 的 instance 是通过 Glance 镜像部署的,所以准备镜像是必须要做的工作.本节介绍 Linux 镜像的制作方法 ...

  2. C语言一维数组转换为二维数组

    一维转二维代码示例: #include <stdio.h> #include <stdlib.h> #define ROW 3 #define COL 2 int main(i ...

  3. 华为荣耀畅玩5C NEM-UL10 ROOT那些事儿(亲测成功)

    以前ROOT手机都是在手机上安装KingRoot 刷机精灵等软件分分钟成功(不排除偶然,,比如这款华为荣耀...) 手机安装KingRoot等软件,,,失败   电脑上安装连接手机Root,,,,失败 ...

  4. 用9种办法解决 JS 闭包经典面试题之 for 循环取 i

    2017-01-06 Tomson JavaScript 转自 https://segmentfault.com/a/1190000003818163 闭包 1.正确的说,应该是指一个闭包域,每当声明 ...

  5. bootstrap 预定义样式风格

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  6. MVC中登录页图片验证码总结

    直接上代码了 using System;using System.Collections.Generic;using System.Drawing;using System.Drawing.Imagi ...

  7. 检查浏览器url改变,处理ajax前进和后退键

    在用ajax获取数据,不刷新页面情况下,保持前进后退按钮功能,网页端兼容性最好的方式如下: 首先url后面参数用#  如http://www.xxx.com/#txf; 使用改变location.ha ...

  8. 《C++ Primer》读书笔记—第二章 变量和基本类型

    声明: 文中内容收集整理自<C++ Primer 中文版 (第5版)>,版权归原书所有. 学习一门程序设计语言最好的方法就是练习编程. 1.8比特的char类型计算机表示的实际范围是-12 ...

  9. 【Java基础】ArrayList工作原理

    ArrayList 以数组实现.节约空间,但数组有容量限制.超出限制时会增加50%容量,用System.arraycopy()复制到新的数组.因此最好能给出数组大小的预估值.默认第一次插入元素时创建大 ...

  10. Asp.net mvc 知多少(十)

    本系列主要翻译自<ASP.NET MVC Interview Questions and Answers >- By Shailendra Chauhan,想看英文原版的可访问http:/ ...