cocos2dx内存管理

原文地址:http://www.cnblogs.com/yu-chao/p/3697459.html

今天想探索一下cocos2dx的内存管理,我们就先从CCObject开始吧

class CC_DLL CCObject : public CCCopying
{
public:
    // object id,CCScriptSupport need public m_uID
    unsigned int        m_uID;
     Lua reference id
    int                 m_nLuaID;
protected:
     count of references
    unsigned int        m_uReference;
     count of autorelease
    unsigned int        m_uAutoReleaseCount;
public:
    CCObject(void);
    virtual ~CCObject(void);
    
    void release(void retain(void);
    CCObject* autorelease(void);
    CCObject* copy(bool isSingleReference(void);
    unsigned int retainCount(virtual bool isEqual(const CCObject* pObject);

    void update(float dt) {CC_UNUSED_PARAM(dt);};
    
    friend class CCAutoreleasePool;
    
    unsigned int getAutoReleaseCount();
};

以上是CCObject的定义,可以看出CCObject主要有两个保护成员:m_uReference和m_uAutoReleaseCount,这个两个成员到底有什么用呢?让我们一探究竟,首先我们来看一下CCObject的构造函数:

CCObject::CCObject(void)
: m_nLuaID(0),m_uReference(1)  when the object is created,the reference count of it is 1,m_uAutoReleaseCount(0)
{
    static unsigned int uObjectCount = 0;

    m_uID = ++uObjectCount;
}
从构造函数我么可以看出m_uReference初始化为1,m_uAutoReleaseCount初始化为0,另外CCObject维护了一个静态无符号整型变量:uObjectCount来记录CCObject实例的总数目。然后我们来看一下release()和retain()函数:

void CCObject::release(void) { if (m_uReference == 0) {modify by yangm CCLOG("CCObject::release reference count should greater than 0"); return; } CCAssert(m_uReference > 0,reference count should greater than 0"); --m_uReference; 0) { delete this; } }
由release()函数可以看到每执行一次release(),m_uReference自减一,只有当m_uReference为零时CCObject对象才会被释放;

void CCObject::retain(CCObject::retain reference count should greater than 0,m_uReference=%d",m_uReference); "); ++m_uReference; }
retain()函数和release()函数功能正好相反,每调用一次retain(),m_uReference会自增一;接下来我们再来看看autorelease()函数:

CCObject* CCObject::autorelease(void)
{
    CCPoolManager::sharedPoolManager()->addObject(this);
    return this;
}
void CCPoolManager::addObject(CCObject* pObject)
{
    getCurReleasePool()->addObject(pObject);  获取栈顶得CCAutoreleasePool实例,并将CCObject加入到CCAutoreleasePool中
}
此处涉及到一个CCPoolManager类,该类内部有一个CCAutoreleasePool类型的栈,而CCPoolManager又是什么呢?其实CCPoolManager就是一个CCArray一个可增长的数组,我们来看看他得addObject(CCObject* pObject)函数:

void CCAutoreleasePool::addObject(CCObject* pObject) { m_pManagedObjectArray->addObject(pObject); 向CCArray中添加CCObject实例 CCAssert(pObject->m_uReference > 1,0); line-height:1.5!important">reference count should be greater than 1"); ++(pObject->m_uAutoReleaseCount); CCObject的m_uAutoReleaseCount自增一 pObject->release(); no ref count,in this case autorelease pool added. }

在调试的时候我们会发现CCObject在release()时他的m_uReference为2,也就避免了被释放,其实该对象实例在添加到自动释放池(CCAutoreleasePool)的时候retain了一下;我们来验证一下,来看看下面的几个函数:

void CCArray::addObject(CCObject* object) { ccArrayAppendObjectWithResize(data,object); } /** Appends an object. Capacity of arr is increased if needed. */ void ccArrayAppendObjectWithResize(ccArray *arr,CCObject* object) { ccArrayEnsureExtraCapacity(arr,1); ccArrayAppendObject(arr,255); line-height:1.5!important">object); } * Appends an object. Behavior undefined if array doesn't have enough capacity. void ccArrayAppendObject(ccArray *arr,255); line-height:1.5!important">object) { CCAssert(object != NULL,0); line-height:1.5!important">Invalid parameter!"); object->retain(); 此处果然retain了一下 arr->arr[arr->num] = object; arr->num++; }
接下来我们来看一下cocos2dx如何销毁那些无效的对象实例,以下是整个程序的主循环:

void CCDisplayLinkDirector::mainLoop(if (m_bPurgeDirecotorInNextLoop) { m_bPurgeDirecotorInNextLoop = false; purgeDirector(); } else if (! m_bInvalid) { drawScene(); release the objects CCPoolManager::sharedPoolManager()->pop(); } }

由上面函数可知coco2dx在每一帧的帧尾会调用自动释放内存池的管理者—CCPoolManager的pop()函数,我们来看看该函数主要做了哪些工作?

void CCPoolManager::pop() { if (! m_pCurReleasePool) { return; } int nCount = m_pReleasePoolStack->count(); m_pCurReleasePool->clear(); if(nCount > 1) { m_pReleasePoolStack->removeObjectAtIndex(nCount-1); if(nCount > 1) { m_pCurReleasePool = m_pReleasePoolStack->objectAtIndex(nCount - 2); return; } m_pCurReleasePool = (CCAutoreleasePool*)m_pReleasePoolStack->objectAtIndex(nCount - 2); } m_pCurReleasePool = NULL;*/ }

主要做的就两点:1、清除当前的自动释放池,2、更新当前自动释放池指针;我们来看看他是如何clear?

void CCAutoreleasePool::clear() { if(m_pManagedObjectArray->count() > 0) { CCAutoreleasePool* pReleasePool; #ifdef _DEBUG int nIndex = m_pManagedObjectArray->count() - 1; #endif CCObject* pObj = NULL; CCARRAY_FOREACH_REVERSE(m_pManagedObjectArray,pObj) { if(!pObj) break; --(pObj->m_uAutoReleaseCount); if( (int)(pObj->m_uAutoReleaseCount)<0 ){ pObj->m_uAutoReleaseCount = 0; } (*it)->release(); delete (*it); #ifdef _DEBUG nIndex--; #endif } m_pManagedObjectArray->removeAllObjects(); } }

以上函数主要做的工作是:1、将内部的所有元素的自动释放引用m_uAutoReleaseCount减一。2、清除内部的所有元素。看到这里我很诧异:为什么要清除所有的元素呢?难道不是清除那些无效的对象实例吗?紧接着看下面的函数看他如何removeAllObjects。

void CCArray::removeAllObjects() { ccArrayRemoveAllObjects(data); } void ccArrayRemoveAllObjects(ccArray *arr) { while( arr->num > 0 ) { (arr->arr[—arr->num])->release(); 注意此处的—arr->num } }

以上函数其实就是将数组内部的所有元素release一下,并将元素个数减为0。到这里整个对象创建—添加到自动释放池—帧尾的释放的过程就完了,我一开始很是纳闷,假如我有定义了一个类如下:

class A : public CCNode { private: CCSprite* m_pSprite1; CCSprite* m_pSprite2; public: CREATE_FUNC(A); bool init(); }; bool A::init() { m_pSprite1 = CCSprite::create(“1.png”); 创建精灵1 m_pSprite2 = CCSprite::create(“2.png”); 创建精灵2 this->addChild(m_pSprite); 只将精灵1添加到父节点 }
实际上到下一帧的时候,m_pSprite2所指向的内存已经无效,m_Sprite1仍然有效。上面的过程是如何做到的呢?只有这种情况才能解释:m_pSprite1和m_pSprite2在create的时候加入到了自动释放池被监视,而m_pSprite1再加入到父节点时retain了一下,才不会在帧尾release的时候被释放掉。我们来验证一下CCNode::addChild(…)(参数就不写了):

void CCNode::addChild(CCNode *child,int zOrder,255); line-height:1.5!important">int tag) { CCAssert( child != NULL,0); line-height:1.5!important">Argument must be non-nil"); CCAssert( child->m_pParent == NULL,0); line-height:1.5!important">child already added. It can't be added again"); if( ! m_pChildren ) { this->childrenAlloc(); } this->insertChild(child,zOrder); child->m_nTag = tag; child->setParent(this); child->setOrderOfArrival(s_globalOrderOfArrival++); if( m_bRunning ) { child->onEnter(); child->onEnterTransitionDidFinish(); } }

相关推荐


    本文实践自 RayWenderlich、Ali Hafizji 的文章《How To Create Dynamic Textures with CCRenderTexture in Cocos2D 2.X》,文中使用Cocos2D,我在这里使用Cocos2D-x 2.1.4进行学习和移植。在这篇文章,将会学习到如何创建实时纹理、如何用Gimp创建无缝拼接纹
Cocos-code-ide使用入门学习地点:杭州滨江邮箱:appdevzw@163.com微信公众号:HopToad 欢迎转载,转载标注出处:http://blog.csdn.netotbaron/article/details/424343991.  软件准备 下载地址:http://cn.cocos2d-x.org/download 2.  简介2.1         引用C
第一次開始用手游引擎挺激动!!!进入正题。下载资源1:从Cocos2D-x官网上下载,进入网页http://www.cocos2d-x.org/download,点击Cocos2d-x以下的Download  v3.0,保存到自定义的文件夹2:从python官网上下载。进入网页https://www.python.org/downloads/,我当前下载的是3.4.0(当前最新
    Cocos2d-x是一款强大的基于OpenGLES的跨平台游戏开发引擎,易学易用,支持多种智能移动平台。官网地址:http://cocos2d-x.org/当前版本:2.0    有很多的学习资料,在这里我只做为自己的笔记记录下来,错误之处还请指出。在VisualStudio2008平台的编译:1.下载当前稳
1.  来源 QuickV3sample项目中的2048样例游戏,以及最近《最强大脑》娱乐节目。将2048改造成一款挑战玩家对数字记忆的小游戏。邮箱:appdevzw@163.com微信公众号:HopToadAPK下载地址:http://download.csdn.net/detailotbaron/8446223源码下载地址:http://download.csdn.net/
   Cocos2d-x3.x已经支持使用CMake来进行构建了,这里尝试以QtCreatorIDE来进行CMake构建。Cocos2d-x3.X地址:https://github.com/cocos2d/cocos2d-x1.打开QtCreator,菜单栏→"打开文件或项目...",打开cocos2d-x目录下的CMakeLists.txt文件;2.弹出CMake向导,如下图所示:设置
 下载地址:链接:https://pan.baidu.com/s/1IkQsMU6NoERAAQLcCUMcXQ提取码:p1pb下载完成后,解压进入build目录使用vs2013打开工程设置平台工具集,打开设置界面设置: 点击开始编译等待编译结束编译成功在build文件下会出现一个新文件夹Debug.win32,里面就是编译
分享一下我老师大神的人工智能教程吧。零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到我们人工智能的队伍中来!http://www.captainbed.net前言上次用象棋演示了cocos2dx的基本用法,但是对cocos2dx并没有作深入的讨论,这次以超级马里奥的源代码为线索,我们一起来学习超级马里奥的实
1. 圆形音量button事实上作者的本意应该是叫做“电位计button”。可是我觉得它和我们的圆形音量button非常像,所以就这么叫它吧~先看效果:好了,不多解释,本篇到此为止。(旁白: 噗。就这样结束了?)啊才怪~我们来看看代码:[cpp] viewplaincopyprint?CCContro
原文链接:http://www.cnblogs.com/physwf/archive/2013/04/26/3043912.html为了进一步深入学习贯彻Cocos2d,我们将自己写一个场景类,但我们不会走的太远,凡是都要循序渐进,哪怕只前进一点点,那也至少是前进了,总比贪多嚼不烂一头雾水的好。在上一节中我们建
2019独角兽企业重金招聘Python工程师标准>>>cocos2d2.0之后加入了一种九宫格的实现,主要作用是用来拉伸图片,这样的好处在于保留图片四个角不变形的同时,对图片中间部分进行拉伸,来满足一些控件的自适应(PS: 比如包括按钮,对话框,最直观的形象就是ios里的短信气泡了),这就要求图
原文链接:http://www.cnblogs.com/linji/p/3599478.html1.环境和工具准备Win7VS2010/2012,至于2008v2版本之后似乎就不支持了。 2.安装pythonv.2.0版本之前是用vs模板创建工程的,到vs2.2之后就改用python创建了。到python官网下载版本2.7.5的,然后
环境:ubuntu14.04adt-bundle-linux-x86_64android-ndk-r9d-linux-x86_64cocos2d-x-3.0正式版apache-ant1.9.3python2.7(ubuntu自带)加入环境变量exportANDROID_SDK_ROOT=/home/yangming/adt-bundle-linux/sdkexportPATH=${PATH}:/$ANDROID_SDK_ROOTools/export
1开发背景游戏程序设计涉及了学科中的各个方面,鉴于目的在于学习与进步,本游戏《FlappyBird》采用了两个不同的开发方式来开发本款游戏,一类直接采用win32底层API来实现,另一类采用当前火热的cocos2d-x游戏引擎来开发本游戏。2需求分析2.1数据分析本项目要开发的是一款游
原文链接:http://www.cnblogs.com/linji/p/3599912.html//纯色色块控件(锚点默认左下角)CCLayerColor*ccc=CCLayerColor::create(ccc4(255,0,0,128),200,100);//渐变色块控件CCLayerGradient*ccc=CCLayerGradient::create(ccc4(255,0,0,
原文链接:http://www.cnblogs.com/linji/p/3599488.html//载入一张图片CCSprite*leftDoor=CCSprite::create("loading/door.png");leftDoor->setAnchorPoint(ccp(1,0.5));//设置锚点为右边中心点leftDoor->setPosition(ccp(240,160));/
为了答谢广大学员对智捷课堂以及关老师的支持,现购买51CTO学院关老师的Cocos2d-x课程之一可以送智捷课堂编写图书一本(专题可以送3本)。一、Cocos2d-x课程列表:1、Cocos2d-x入门与提高视频教程__Part22、Cocos2d-x数据持久化与网络通信__Part33、Cocos2d-x架构设计与性能优化内存优
Spawn让多个action同时执行。Spawn有多种不同的create方法,最终都调用了createWithTwoActions(FiniteTimeAction*action1,FiniteTimeAction*action2)方法。createWithTwoActions调用initWithTwoActions方法:对两个action变量初始化:_one=action1;_two=action2;如果两个a
需要环境:php,luajit.昨天在cygwin上安装php和luajit环境,这真特么是一个坑。建议不要用虚拟环境安装打包环境,否则可能会出现各种莫名问题。折腾了一下午,最终将环境转向linux。其中,luajit的安装脚本已经在quick-cocos2d-x-develop/bin/中,直接luajit_install.sh即可。我的lin
v3.0相对v2.2来说,最引人注意的。应该是对触摸层级的优化。和lambda回调函数的引入(嗯嗯,不枉我改了那么多类名。话说,每次cocos2dx大更新。总要改掉一堆类名函数名)。这些特性应该有不少人研究了,所以今天说点跟图片有关的东西。v3.0在载入图片方面也有了非常大改变,仅仅只是