游戲AI(二)—行為樹優化之記憶體優化

来源:http://www.cnblogs.com/moonmagician/archive/2017/12/24/8099327.html
-Advertisement-
Play Games

上一篇我們講到了AI架構之一的行為樹,本篇文章和下一篇文章我們將對行為樹進行優化,在本篇文章中我們講到的是記憶體優化 問題 上一篇中我們設計的行為樹由於直接採用new進行動態記憶體分配,沒有自己進行管理。因此行為樹各節點的存儲位置會散佈在記憶體空間的各處,行為樹在不同節點中切換時會導致Cache頻繁失效。 ...


上一篇我們講到了AI架構之一的行為樹,本篇文章和下一篇文章我們將對行為樹進行優化,在本篇文章中我們講到的是記憶體優化

問題

上一篇中我們設計的行為樹由於直接採用new進行動態記憶體分配,沒有自己進行管理。因此行為樹各節點的存儲位置會散佈在記憶體空間的各處,行為樹在不同節點中切換時會導致Cache頻繁失效。
通過記憶體管理改變行為樹節點的記憶體分佈,可以顯著提高行為樹的記憶體性能。

解決辦法

我們可以在BehaviorTree中引入一組記憶體分配的API來保證各節點儘量分配在連續的記憶體上,代碼如下

BehaviorTree(Behavior*InRoot):Root(InRoot),
Buffer(new uint8_t[MaxBehaviorTreeMemory]),Offset(0){}
~BehaviorTree(){ delete[] Buffer; }
    
template<typename T>
        T* Allocate()
        {
            T* Node = new((void*)((uintptr_t)Buffer + Offset)) T;
            Offset += sizeof(T);
            assert(Offset < MaxBehaviorTreeMemory);
            return Node;
        }

我們在BehaviorTree中引入一個Allocate函數用來負責所有節點的記憶體分配。
當行為樹被構造時,一塊用於保存節點的記憶體空間Bufffer會隨之分配,Allocate函數通過Placement new在Buffer上進行記憶體分配,通過Offset記錄分配已分配記憶體的偏移地址。
通過這種方式我們可以讓節點分配在連續的記憶體上,同時通過控制分配節點的順序(如深度遍歷廣度遍歷等),我們可以進一步減少行為樹遍歷時產生的Cache失效,提高記憶體性能。)

複合節點

除了對節點分配進行優化,我們還可以改變複合節點的記憶體佈局,進一步提升性能。

class Composite :public Behavior
    {
    public:
        friend class BehaviorTree;
        virtual void AddChild(Behavior* InChild) override
        { 
            assert(ChildrenCount < MaxChildrenPerComposite);
            ptrdiff_t p = (uintptr_t)InChild - (uintptr_t)this;
            assert(p < std::numeric_limits<uint16_t>::max());
            Children[ChildrenCount++] = static_cast<uint16_t>(p);
        }   

        Behavior* GetChild(size_t index)
        {
            assert(index < MaxChildrenPerComposite);
            return (Behavior*)((uintptr_t)this + Children[index]);
        }

        size_t GetChildrenCount()
        {
            return ChildrenCount;
        }

protected:
        uint16_t Children[MaxChildrenPerComposite];
        uint16_t ChildrenCount = 0;
    };

在如上代碼中,我們通過靜態數組代替vector,避免在存儲時vector所產生的額外堆操作,通過用保存子節點相對於複合節點的偏移地址來代替直接保存子節點指針以節省記憶體空間。由於更換了子節點的存儲方式,我們需要通過getchild()函數來根據複合節點地址和子節點偏移地址獲得子節點指針。

總結

以上,就是關於行為樹的記憶體優化方式,當然凡事無絕對,究竟如何構造行為樹應當根據實際情況選擇,下一篇我們將講述另一種行為樹優化方法。
[gihub鏈接][1]


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 架構的核心就是 交互,而實現這個交互的前提是弄清楚H5和Native端的交互 本文主要介紹Native端(Android/iOS)和H5端(泛指前端)的交互原理 (之前也整理過類似的文章,本系列重新梳理) 與`H5`交互的兩種方式 原生和前端的交互有兩種方式: 以及 (在Android中是 ) ...
  • jquery中ajax請求後臺數據成功後既不執行success也不執行error,此外系統報錯:Uncaught SyntaxError: Unexpected identifier at Object.success,但後臺能夠返回數據,原代碼如下: 主要原因在於後臺返回的數據並非json格式,而 ...
  • 國際化環境下系統架構演化資料庫架構Related Link:JFrog Artifactory https://jfrog.com/open-source/Ansible http://www.ansible.com.cn/index.htmlgrafana https://grafana.com/... ...
  • 線上聊天項目結構圖: 多用戶登陸效果圖: 多用戶聊天效果圖: 資料庫效果圖: 重新構建了Server類,使用了Gson方法,通過解析Json字元串,增加Info類,簡化判斷過程。 Server類代碼如下: 新增的工具類Info,通過info對象的get方法把獲得服務端得到的各種字元串直接歸類了。 I ...
  • java二分查找法 今晚上在逛碼雲的時候,看到一個php的二分查找法,看著這詞比較熟悉,(因為當時學php的時候也學過,只是工作中沒用過,又加上學的時候就稀里糊塗的,慢慢的就忘了),就點進去了,就用java寫了一遍,使用的遞歸。 二分查找法的個人理解: 首先你得是個從小到大順序的數組,進行數組索引取 ...
  • 建模的重要性: 為了更好的理解一個系統 管理複雜度(也就是為了敏捷開發) 理解不同的關註點 儘早發現問題和遺漏 與項目干係人進行溝通 驅動實現 資源高效利用 ...
  • 簡單聊天Demo 使用tcp協議實現的簡單聊天功能(非常簡單的) 思想:使用2個線程,一個線程是用來接收消息的,另一個線程是用來發消息的。 客戶端Demo代碼: 伺服器端Demo代碼: 發送線程的Demo代碼: 接收線程的Demo代碼: ...
  • 今天的學習內容是python中的列表的相關內容。 一.創建列表 1.創建一個普通列表 2.創建一個混合列表 3.創建一個空列表 三種方式就介紹給大家了,接下來,如果想向列表中添加元素,該怎麼辦呢? 二.向列表中添加元素 1.append >>> tabulation1.append('紫霞')>>> ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...