前端之家收集整理的这篇文章主要介绍了
cocos2dx的内存管理机制,
前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
必须尊重作者:
版权声明:本文由(小塔)原创,转载请保留文章出处!
本文链接:http://www.zaojiahua.com/memory-management.html
今天看了一下cocos2dx的内存管理机制,有些地方不太好理解搞了挺长的时间,现在感觉自己理解的差不多了,赶快写下自己的思路和互联网的广大朋友分享,如果你发现有错误的地方或者不理解的地方欢迎指正!首先我们必须说一下c++中变量的内存空间的分配问题,我们在c++中写一个类,可以在栈上分配内存空间也可以使用new在堆上分配内存空间,如果类对象是在栈上分配的内存空间,这个内存空间的管理就不是我们的事了,但如果是在堆上分配的内存空间,当然需要我们来手动的delete了!cocos2dx采用的是在堆上分配内存空间,想想看你在写程序的时候对于cocos2dx中的类是不是大多数都是通过工厂方法获得的一个指针,你见过在栈上分配内存空间的情况吗?所以问题来了,既然在堆上分配内存空间,那么如何管理这个内存空间,什么时候应该释放就是个问题了!在程序中,当我们创建了一个对象的时候,这块内存空间经常是被不同的对象引用,如果删除的早了,有对象还在引用这块内存空间那么程序必然要崩溃!所以cocos2dx引入了引用计数这个内存管理机制。
当我们在堆上分配一块内存空间的时候,这个对象的引用计数就是1,当有对象要引用这块内存空间的时候,这个引用计数就增加1,当有对象不再引用这块内存的时候引用计数就减1,当这个引用计数减为0的时候就使用delete删除掉这块内存,这样就做到了当有对象引用的时候会正常的访问这块内存,引用完毕也可以正常的回收。先来看一下如下的代码,有关引用计数的问题就会很清楚了!
3 |
//CCObject::CCObject(void) |
4
//,m_uReference(1) {}// when the object is created,the reference count of it is 1 |
5 |
CCSprite * sprite = new CCSprite(); |
6
|
7 |
CCLog("retain count:%d" ,sprite->retainCount()); |
10
sprite->retain(); |
11 |
12 |
13 |
//调用release方法的时候引用计数减一,当这个引用计数减为0的时候,在release方法中会delete掉这个对象 |
14
sprite->release(); |
18
//CCPoolManager::sharedPoolManager()->addObject(this); |
19 |
//调用autorelease方法的时候对象会被放到自动回收池中,这个自动回收池在每帧结束的时候会调用一次对象的release方法 |
20
sprite->autorelease(); |
21 |
上边的代码有一处不是很好的理解,就是大家经常说的自动回收机制,也就是上边的autorelease方法,这个方法到底为我们做了什么,有什么作用,我们需要好好的搞清楚!首先需要澄清的一个概念就是帧,我们经常的说每秒多少多少帧,其实这个帧需要多少时间不是固定的,这个需要看每帧我们需要做多少事情,如果没一帧我们需要渲染很多的东西,那这一帧执行的时间当然就会很长的,游戏显得就会很卡,这个时候每秒的帧率就会下降的,所以不是时间决定的帧率,而是帧影响的时间!如果用户不手动创建autoreleasepool,只会存在一个default cocos2d autorelease pool在栈底,这个自动回收池就是在每帧结束的时候起作用的,在游戏的每一帧都会有一个大的循环,在这一帧的过程中,当我们调用了autorelease方法以后,我们的对象就会放到这个内存回收池中,当一帧结束的时候这个内存回收池就会释放掉,这个时候在内存回收池中的对象就会被release一下,也就是说引用计数就会减一,如果这个时候引用计数为0,就会删除对象了。如果引用计数不为0的话对象是不会被删除的,下一帧开始的时候系统又会创建一个内存回收池,这个时候在上一次添加的对象这个时候是不会重新添加到这个内存回收池中的,在这个内存回收池中的对象是你在这一帧中调用了autorelease函数的对象。好了,内存回收池我觉的我已经说清楚了。下面我们来看一下,通常我们的代码都是怎么写的,来看看这个自动回收机制怎么就做到自动回收了!
//在create的时候 调用了sprite的autorelease 方法 |
CCSprite * sprite = CCSprite::create(
"HelloWorld.png"
);
this
->addChild(sprite);
首先我们需要分析一下上边的代码,调用了create工厂方法以后,内部的实现是先new一个CCSprite的对象,这个时候引用计数加1,然后调用autorelease方法,将这个对象放到了自动回收池中,因为这一帧还没有结束,当然引用计数就还是1,所以打印的结果就是1,当我们调用addChild的时候,传入这个CCSprite对象,这个时候在当前层接受了这个对象以后会把它的引用计数加一,表明当前层正在使用这块内存空间,所以现在的retain就是2了。当这一帧结束的时候自动回收池会将对象的引用计数-1,所以现在就只有CCLayer在引用这个对象了,当CCLayer析构的时候,它会调用这个对象的release方法,这个时候当然就会删除了这个CCSprite对象了。所以什么是自动回收机制呢,自动就是在这一帧结束的时候将对象开始new的时候加的那个引用计数减掉,而让引擎中持有对象引用的其他类去管理这个对象,当持有者析构的时候就删除引用,引擎中的类负责retain和release,这个也算是自动吧!下面我们通过俩个例子来理解一下这个内存管理机制。
bool
HelloWorld::init()
{
if
( !CCLayer::init() )
returnfalse
;
CCMenuItemImage *pCloseItem = CCMenuItemImage::create(
"CloseNormal.png"
"CloseSelected.png"menu_selector(HelloWorld::menuCloseCallback)); |
CCMenu* pMenu = CCMenu::create(pCloseItem,NULL);
->addChild(pMenu,1);
->m_sprite = CCSprite::create();
"%d"->m_sprite->retainCount());
25 |
voidHelloWorld::menuCloseCallback(CCObject* pSender) |
26
27 |
->m_sprite->retainCount()); |
28
->m_sprite->getPosition(); |