cocos2dx中的retain和release
原文地址:http://www.jb51.cc/article/p-zlltzegt-zw.html
自学使用
1.为什么会有retain?
C++和Java不一样,Java有一套很方便的垃圾回收机制,当我们不需要使用某个对象时,给它赋予null值即可。而C++new了一个对象之后,不使用的时候通常需要delete掉。
于是,Cocos2d-x就发明了一套内存管理机制(小若:发你妹纸。。。),其实红孩儿的博客很详细地解释了Cocos2d-x的内存管理机制,我没有能力也不想重复解释。(小若:那你还写?==)
Retain的意思是保持引用,也就是说,如果想保持某个对象的引用,避免它被Cocos2d-x释放,那就要调用对象的retain函数。(小若:为什么不retain就会被释放?)
2.真正的凶手autoRelease
既然旁白诚心诚意地问我,那我就光明正大地回答吧(小若:我今天没力气吐槽,好吧==)。
一旦调用对象的autoRelease函数,那么这个对象就被Cocos2d-x的内存管理机制给盯上了,如果这个对象没人认领,那就等着被释放吧。(小若:==太久没吐槽,一时不知道吐什么好)。
3.看代码实际点
说了这么多,还是上代码吧。
创建一个Cocox2d-x的项目,就直接拿HelloWorldScene开刀,修改init函数,在最后添加一句代码:
- boolHelloWorld::init()
- {
- boolbRet=false;
- do
- {
-
-
- testSprite=CCSprite::create("HelloWorld.png");
- bRet=true;
- }while(0);
-
- returnbRet;
- }
(小若:testSprite是什么小编?)
testSprite是一个成员变量,在头文件里加上就可以了:
classHelloWorld:publiccocos2d::cclayer
- public:
- virtualboolinit();
- staticcocos2d::CCScene*scene();
- voidmenuCloseCallback(CCObject*pSender);
- CREATE_FUNC(HelloWorld);
- private:
- cocos2d::CCSprite*testSprite;
- };
然后,最关键的来了,我们修改menuCloseCallback函数:
voidHelloWorld::menuCloseCallback(CCObject*pSender)
- testSprite->getPosition();
- }
现在,运行项目,点击按钮,看看是什么情况?
(小若:报错了!)
如果大家知道怎么调试项目的话,我们在menuCloseCallback函数里断点,用调试模式运行项目,看看testSprite对象:
(小若:很正常啊,有什么特别的?)
正你妹纸啊,正!你才正!(小若:不要这么光明正大地赞我OO!)
我们应该能看到不少非正常数据,图中已经用红色圈圈标出来了,这代表testSprite对象被释放了,现在testSprite指向未知的位置。
这是很危险的,有时候它不会立即报错,但是在某个时刻突然崩溃!
要想解决这个问题,很简单,再次修改init函数:
testSprite->retain();
-
- bRet=true;
- }while(0);
- returnbRet;
- }
再次运行项目,看看还会不会报错?(小若:不会了,为什么?)
再次用调试模式运行项目,看看testSprite对象:
(小若:不正常!都是0!!)
零你妹纸==(小若:为什么今天你总是抢我的对白OO!)
这次我们看到testSprite的数据明显正常了。
4.原理来了
好了,唠叨了一大堆,还没有进入正题。
首先,要想让对象参与内存管理机制,必须继承CCObject类(CCNode、cclayer等都继承了CCObject类)。
然后,调用对象的autoRelease函数,对象就会被Cocos2d-x的内存管理机制盯上,在游戏的每一帧,内存管理机制都会扫描一遍被盯上的对象,一旦发现对象无人认领,就会将对象杀死!(小若:嗷~残忍!)
如果不想让对象被杀死,那么就要调用对象的retain函数,这样对象就被认领了,一旦对象被认领,就永远不会被内存管理机制杀掉,是永远,一辈子。(小若:好朋友,一辈子==)
但,对象一辈子都不被释放的话,那么就会产生内存泄露,你试试加载一个占20M内存的对象一辈子不释放,不折腾死才怪~(小若:你去加载一个20M的对象本身就是闲的那个什么疼啊!)因此,当你不需要再使用这个对象时,就要调用对象的release函数,这是和retain对应的。一般可以在析构函数里调用release函数。
5.实际情况
讲道理,大家都懂,但是,相信很多朋友在实际写代码的时候,还是会感觉很混乱。
比如,什么时候该retain?大家是不是发现,有时候不retain也不会报错?
其实这很简单,因为我们经常会在create一个对象之后,添加到层里,如:
testSprite=CCSprite::create("HelloWorld.png");
this->addChild(testSprite);
addChild函数就是导致大家混乱的凶手了,addChild函数会调用对象的retain函数,为什么它要调用对象的retain函数呢?因为你都把对象送给它当孩子了,它当然要认领这个对象了!(小若:我懂了,嗷!)
于是,当我们把对象addChild到cclayer时(不一定是cclayer,CCArray、CCNode都行),我们就不需要调用对象的retain函数了。
6.那倒底什么时候要retain?
说了这么多,还是没有说清楚,什么时候要调用对象的retain。
很简单,当你把一个对象作为成员变量时,并且没有把对象addChild到另外一个对象时,就需要调用retain函数。
7.最后的最后
一定要记住,必须要调用了对象的autoRelease函数之后,retain和release函数才会生效,否则,一切都是徒劳。
因此,十分建议使用create的方式创建对象,如:
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。