• 【转载】ogre内存管理


    原文:ogre内存管理

    OGRE内存分配策略相关文件及简述

    OGRE提供了自己的内存分配策略,甚至为STL容器提供了新的分配策略,相关文件及简述如下:

    OgreMemoryAllocatedObject.h  OgreMemoryAllocatedObject.cpp

    // 所有使用Ogre内存分配器的类的父类

    OgreMemoryAllocatorConfig.h

    // 配置内存分配相关规则

    OgreMemoryNedAlloc.h  OgreMemoryNedAlloc.cpp

    // 使用nedalloc库,定义了类NedAllocPolicy、NedAlignedAllocPolicy

    OgreMemoryNedPooling.h  OgreMemoryNedPooling.cpp

    // 使用nedalloc库,定义了类NedPoolingPolicy、NedPoolingAlignedPolicy

    OgreMemoryStdAlloc.h

    // 定义了类StdAllocPolicy、StdAlignedAllocPolicy

    // 只是对malloc/free的简单包装

    OgreMemorySTLAllocator.h

    // 为STL容器提供的分配器

    OgreMemoryTracker.h  OgreMemoryTracker.cpp

    // 用于跟踪内存的分配和释放,并统计内存的使用和泄漏情况

    // *为避免循环引用,OgreMemoryTracker.h头文件必须在OgrePrerequisites.h引用后引用*

    OgreAlignedAllocator.h  OgreAlignedAllocator.cpp

    // 提供对齐内存分配函数

    对于OGRE中STL容器的内存分配策略的简述:

    以下是摘自OgrePrerequisites.h的代码:

    template <typename T, typename A = STLAllocator<T, GeneralAllocPolicy> >

    struct vector

    {

    #if OGRE_CONTAINERS_USE_CUSTOM_MEMORY_ALLOCATOR

         typedef typename std::vector<T, A> type;   

    #else

         typedef typename std::vector<T> type;   

    #endif

    };

    如上所示,OGRE对STL容器进行了简单的包装,使其默认使用OGRE提供了内存分配策略GeneralAllocPolicy。在

    OgreMemoryAllocatorConfig.h头文件中,能够容易找到GeneralAllocPolicy的定义。这样,在使用STL容器时须加“::type”,如Ogre::vector<T>::type,或者照旧使用STL本身的内存分配策略,如std::vector<T>。如需要改变STL容器使用的内存策略,则可以按如下方式使用,Ogre::vector<T,STLAllocator<T,CustomPolicy>>::type,其中CustomPolicy为用户指定的内存策略。

    OGRE的内存策略配置及如何自定义内存策略

    OGRE的内存策略配置

    OGRE中默认所有类型使用一致的内存分配策略,并在编译时便决定。在OgreMemoryAllocatorConfig.h文件中由OGRE_MEMORY_ALLOCATOR宏选择相应的内存策略。

    OGRE提供了如下三种内存策略:

        NedAllocPolicy(NedAlignedAllocPolicy)

        NedPoolingPolicy(NedPoolingAlignedPolicy)

        StdAllocPolicy(StdAlignedAllocPolicy)

    在OgreConfig.h中定义有

        #define OGRE_MEMORY_ALLOCATOR_STD 1

        #define OGRE_MEMORY_ALLOCATOR_NED 2

        #define OGRE_MEMORY_ALLOCATOR_NEDPOOLING 4

    在OgreBuildSettings.h中定义有

        #define OGRE_MEMORY_ALLOCATOR 4

    即OGRE默认使用的内存分配策略为NedPoolingPolicy。

    *** OgreMemoryAllocatorConfig.h文件中定义MemoryCategory枚举类型,以支持不同目的使用不同的内存分配方式,但需用户自己实现。

    如何自定义内存分配策略

    (假定类名为CustomAllocPolicy,对齐类为CustomAlignAllocPolicy):

    1. 在CustomAllocPolicy和CustomAlignAllocPolicy中至少实现如下三个静态函数:

    //分配内存

    static inline void* allocateBytes(size_t count, const char* file = 0, int line = 0, const char* func = 0);

    //释放内存

    static inline void deallocateBytes(void* ptr);

    //获取一次性分配的最大分配数量

    static inline size_t getMaxAllocationSize();

    2. 在OgreConfig.h头文件中添加相应宏定义

    如  #define OGRE_MEMORY_ALLOCATOR_CUSTOM 5

    3. 在OgreMemoryAllocatorConfig.h头文件中添加代码行,如

    #elif OGRE_MEMORY_ALLOCATOR == OGRE_MEMORY_ALLOCATOR_CUSTOM

    # include "OgreMemoryCustomAlloc.h"  //CustomAllocPolicy类所在头文件

    namespace Ogre

    {

    template <MemoryCategory Cat> class CategorisedAllocPolicy : public CustomAllocPolicy{};

    template <MemoryCategory Cat, size_t align = 0> class CategorisedAlignAllocPolicy : public CustomAlignAllocPolicy<align>{};

    }

    4. 最后在OgreBuildSettings.h中改变定义

    #define OGRE_MEMORY_ALLOCATOR 


    如何在程序中使用OGRE提供的内存策略

    要使用OGRE提供的内存策略(或自定义的内存策略)主要有两种方式:

    1. 继承AllocatedObject类

    2. 使用OgreMemoryAllocatorConfig.h头文件中定义的内存操作宏

    继承AllocatedObject类

    AllocatedObject类中重载了new、new[]、delete及delete[]等操作符,所以若要A类使用Ogre提供的或自定义的内存策略,只需继承AllocatedObject类即可。AllocatedObject类以模板的形式支持各种内存分配策略。在OgreMemoryAllocatorConfig.h头文件中对各种模板的AllocatedObject类进行了“重命名”,例如:

    typedef CategorisedAllocPolicy<Ogre::MEMCATEGORY_GENERAL> GeneralAllocPolicy;

    typedef AllocatedObject<GeneralAllocPolicy> GeneralAllocatedObject;

    typedef GeneralAllocatedObject  ArchiveAlloc;

    typedef GeneralAllocatedObject  ConfigAlloc;

    所以在继承使用AllocatedObject类时,可考虑使用OgreMemoryAllocatorConfig.h头文件中的宏定义。例如:

    (举例多取自OGRE源码,一般都标注了其文件名)

    class _OgreExport ConfigFile : public ConfigAlloc //OgreConfigFile.h

    class _OgreExport ArchiveManager : public Singleton<ArchiveManager>, public ArchiveAlloc  // OgreArchiveManager.h

    而为了统一形式,在OgreMemoryAllocatorConfig.h头文件中为继承自AllocatedObject的类定义如下两个宏:

    #define OGRE_NEW new

    #define OGRE_DELETE delete

    因此在使用时,形式如下

    ArchiveManager* mArchiveManager = OGRE_NEW ArchiveManager();

    使用内存操作宏

    除了继承使用AllocatedObject类之外,对于C++中的原始类型(int, double等),或者来自外部库的不能更改的类型,或者某些原因而不能继承AllocatedObject的类型来说,如果要对它们使用Ogre自定义内存策略,则可以使用OgreMemoryAllocatorConfig.h头文件中定义的内存操作宏:

    以下三个宏用于分配原始内存,只是分配相应大小的内存,并不进行初始化:

    OGRE_MALLOC(bytes, category)

           -- 分配bytes大小的内存

    OGRE_ALLOC_T(T, count, category)

           -- 分配sizeof(T) * count大小的内存

    OGRE_FREE(ptr, category)

             -- 与上述两个配对使用,释放ptr指向的内存

    以下四个宏分配相应内存并进行初始化:

    OGRE_NEW_T(T, category)

            -- 分配sizeof(T)大小的内存,并调用T的构造函数

    OGRE_NEW_ARRAY_T(T, count, category)

            -- 分配sizeof(T) * count大小的内存,并对count个T类型实例依次初始化

    OGRE_DELETE_T(ptr, T, category)

            – 与OGRE_NEW_T 配对使用,调用T的析构函数,释放相应内存

    OGRE_DELETE_ARRAY_T(ptr, T, count, category)

            --与OGRE_NEW_ARRAY_T配对使用,对count个T类型实例依次调用其析构函数,释放相应内存

    举例:

    // OgreTexture.cpp

    void* pixData = OGRE_MALLOC(dataSize, Ogre::MEMCATEGORY_GENERAL);

    // OgreAxisAlignedBox.h

    Vector3* mpCorners = OGRE_ALLOC_T(Vector3, 8, MEMCATEGORY_SCENE_CONTROL);

    OGRE_FREE(mpCorners, MEMCATEGORY_SCENE_CONTROL);

    // OgreAnimationState.cpp

    BoneBlendMask* mBlendMask = OGRE_NEW_T(BoneBlendMask, MEMCATEGORY_ANIMATION)(blendMaskSizeHint);

    OGRE_DELETE_T(mBlendMask, BoneBlendMask, MEMCATEGORY_ANIMATION);

    // OgreBspLevel.cpp

    Brush * mBrushes = OGRE_NEW_ARRAY_T(BspNode::Brush, mNumBrushes, MEMCATEGORY_GEOMETRY);

    OGRE_DELETE_ARRAY_T(mBrushes, Brush, (size_t)mNumBrushes, MEMCATEGORY_GEOMETRY);

    对于以上7个内存操作宏分别有与其对应的对齐方式的宏定义,如OGRE_MALLOC_SIMD、OGRE_MALLOC_ALIGN、OGRE_ALLOC_T_SIMD、OGRE_NEW_T_ALIGN、OGRE_DELETE_ARRAY_T_ALIGN 等等。

    ***使用这些宏时,如果类型T本身继承于AllocatedObject,对程序本身也不会产生什么坏的影响,除了显得多此一举。

    Ogre提供的内存分配/释放跟踪器

    在Ogre自己提供的NedAllocPolicy(NedAlignedAllocPolicy)、NedPoolingPolicy(NedPoolingAlignedPolicy)、StdAllocPolicy(StdAlignedAllocPolicy)三种内存分配策略中,都包含了分配/释放跟踪功能,如需在自定义的分配策略中添加跟踪功能,照搬即可。该功能由MemoryTracker类实现,它统计内存的分配和释放情况,以及内存分配语句所在文件名、行号和函数名。默认在程序结束时将统计信息输出到终端,并保存于OgreLeaks.log文件,也可通过成员函数setReportToStdOut(bool rep)和setReportFileName(const std::string& name)设置输出方式。

    跟踪器的使用方式都在分配策略中完成,无需在它处额外操作。唯一需要设置的就是跟踪器开关的打开与关闭

    在OgrePrerequisites.h文件中有如下宏定义

    #if OGRE_DEBUG_MODE

    #if OGRE_MEMORY_TRACKER_DEBUG_MODE

    #define OGRE_MEMORY_TRACKER 1

    #else

    #define OGRE_MEMORY_TRACKER 0

    #endif

    #else

    #if OGRE_MEMORY_TRACKER_RELEASE_MODE

    #define OGRE_MEMORY_TRACKER 1

    #else

    #define OGRE_MEMORY_TRACKER 0

    #endif

    #endif

    即,OGRE_MEMORY_TRACKER宏开关取决于OGRE_MEMORY_TRACKER_DEBUG_MODE 或OGRE_MEMORY_TRACKER_RELEASE_MODE。而在OgreBuildSettings.h中定义有

    #define OGRE_MEMORY_TRACKER_DEBUG_MODE 0

    #define OGRE_MEMORY_TRACKER_RELEASE_MODE 0

    即默认是关闭跟踪器的,如果需要在DEBUG版下打开跟踪器,只需要将OGRE_MEMORY_TRACKER_DEBUG_MODE设置为1。而根据OgreMemoryAllocatorConfig.h文件中内存操作宏的设置():

    #if OGRE_DEBUG_MODE    //语句1

    …//详见源文件

    #else // !OGRE_DEBUG_MODE

    #endif // OGRE_DEBUG_MODE

    即便是打开了OGRE_MEMORY_TRACKER_RELEASE_MODE宏,跟踪器也不能记录内存分配语句所在文件名、行号和函数名,即不能正常工作。如果需要在Release版下正常开启跟踪器功能,建议将语句1“#if OGRE_DEBUG_MODE”改为“#if OGRE_MEMORY_TRACKER”。

    举例:

    对于如下代码

    #include <string>

    #include <OgreRoot.h>

    #include <OgreMemoryAllocatorConfig.h>

    #include <OgreArchiveManager.h>

    using std::string;

    int main()

    {

    {

        int *pi = OGRE_NEW_T(int, Ogre::MEMCATEGORY_GENERAL)(50);

        OGRE_FREE(pi, Ogre::MEMCATEGORY_GENERAL); //释放

        double *pd = OGRE_NEW_ARRAY_T(double, 5, Ogre::MEMCATEGORY_GENERAL);

        OGRE_FREE(pd, Ogre::MEMCATEGORY_GENERAL); //释放

        string *ps =  OGRE_NEW_T(string, Ogre::MEMCATEGORY_GENERAL)("hello");

        OGRE_DELETE_T(ps, string, Ogre::MEMCATEGORY_GENERAL); //释放

    }

    {

        Ogre::ArchiveManager* mArchiveManager = OGRE_NEW Ogre::ArchiveManager();

        OGRE_DELETE mArchiveManager; //释放

    }

    return 0;

    }

    现开启跟踪器宏开关,即将OGRE_MEMORY_TRACKER_DEBUG_MODE 和 OGRE_MEMORY_TRACKER_RELEASE_MODE 都置为1(须重新编译)。该代码正确释放了所有内存,不存在内存泄漏。如果将代码中释放内存的语句全部注释掉,则在Debug版下生成的OgreLeaks.log文件的内容为:

    Ogre Memory: Detected memory leaks !!!

    Ogre Memory: (6) Allocation(s) with total 244 bytes.

    Ogre Memory: Dumping allocations ->

    e:vs projectogretest estmain.cpp(12) : {4 bytes} function: main

    e:vs projectogretest estmain.cpp(15) : {40 bytes} function: main

    e:vs projectogretest estmain.cpp(25) : {64 bytes} function: main

    (unknown source):(0) : {52 bytes} function:

    (unknown source):(0) : {52 bytes} function:

    e:vs projectogretest estmain.cpp(18) : {32 bytes} function: main

    在Release版下生成的OgreLeaks.log文件的内容为:

    Ogre Memory: Detected memory leaks !!!

    Ogre Memory: (8) Allocation(s) with total 248 bytes.

    Ogre Memory: Dumping allocations ->

    (unknown source):(0) : {48 bytes} function:

    (unknown source):(0) : {28 bytes} function:

    (unknown source):(0) : {4 bytes} function:

    (unknown source):(0) : {4 bytes} function:

    (unknown source):(0) : {4 bytes} function:

    (unknown source):(0) : {72 bytes} function:

    (unknown source):(0) : {40 bytes} function:

    (unknown source):(0) : {48 bytes} function:

    可以看出在Debug版下,跟踪器正常工作,顺利的检测出全部四处内存泄漏,而在Release版下则不能正常工作。将“#if OGRE_DEBUG_MODE”改为“#if OGRE_MEMORY_TRACKER”后,重新在Release版下测试,跟踪器便能正常工作,生成的OgreLeaks.log的内容为:

    Ogre Memory: Detected memory leaks !!!

    Ogre Memory: (8) Allocation(s) with total 248 bytes.

    Ogre Memory: Dumping allocations ->

    (unknown source):(0) : {48 bytes} function:

    .TestMain.cpp(18) : {28 bytes} function: main

    .TestMain.cpp(12) : {4 bytes} function: main

    (unknown source):(0) : {4 bytes} function:

    (unknown source):(0) : {4 bytes} function:

    .TestMain.cpp(25) : {72 bytes} function: main

    .TestMain.cpp(15) : {40 bytes} function: main

    (unknown source):(0) : {48 bytes} function:

    P.S.

    1. 测试代码中,对int和double类型使用的是OGRE_NEW_T 和 OGRE_FREE 来分别分配和释放内存,而并非前文所说的OGRE_DELETE_T,原因是因为对于int,double等基本类型来说,是没有析构函数的,而OGRE_DELETE_T宏是要调用相应析构函数的。

    2. 测试代码中,若将using std::string;注释掉,并将相应string类型测试语句换为如下两句:

    std::string *ps2 = OGRE_NEW_T(std::string, Ogre::MEMCATEGORY_GENERAL)("world");

    OGRE_DELETE_T(ps2, basic_string, Ogre::MEMCATEGORY_GENERAL);

    其中basic_string 不能是string 或 std::string !!!  同样是因为OGRE_DELETE_T宏要调用basic_string类的析构函数。类似情况还有在使用std::fostream,std::fistream等等时。

     
     
  • 相关阅读:
    【Python】练习题
    Markdown 常见用法
    程序员个性注释图案
    Class.getResource和ClassLoader.getResource的区别分析
    spring源码分析(二)
    spring源码分析(一)
    加密、解密、公钥,私钥、Https协议
    二叉查找树、红黑树
    HashMap、ConcurrentHashMap
    springIOC
  • 原文地址:https://www.cnblogs.com/zhehan54/p/5656911.html
Copyright © 2020-2023  润新知