cocos2dx[3.2](24)――内存管理机制

【唠叨】

整合参考文档。


【参考】

http://zh.wikipedia.org/wiki/引用计数(引用计数――维基百科)

http://cn.cocos2d-x.org/tutorial/show?id=2300(引用计数和自动释放池)

http://cn.cocos2d-x.org/tutorial/show?id=1331(内存管理――绕不过去的坎)

http://www.jb51.cc/article/p-qcvudonu-ow.html(内存优化)

https://github.com/chukong/cocos-docs/blob/master/manual/framework/native/v3/memory-management/zh.md(内存管理机制)



【内存管理机制】

在3.x版本,Cocos2d-x采用全新的根类Ref,实现Cocos2d-x 类对象的引用计数记录。引擎中的所有类都派生自Ref。


1、引用计数

引用计数的概念参考《维基百科》:http://zh.wikipedia.org/wiki/引用计数


Cocos2d-x 提供引用计数管理内存。

>调用 retain() 方法 :令其引用计数增1,表示获取该对象的引用权。

> 调用 release() 方法 :在引用结束的时候,令其引用计数值减1,表示释放该对象的引用权。

> 调用 autorelease() 方法 :将对象放入自动释放池。

>当释放池自身被释放的时候,它就会对池中的所有对象执行一次release()方法,实现灵活的垃圾回收。

Cocos2d-x 提供 AutoreleasePool,管理自动释放对象。

>当释放池自身被释放的时候,它就会对池中的所有对象执行一次release()方法。


核心类Ref:实现了引用计数。

//
/**
*	CCRef.h
**/
	classCC_DLLRef
	{
		public:
			voidretain();//保留。引用计数+1
			voidrelease();//释放。引用计数-1
			Ref*autorelease();//实现自动释放。
			unsignedintgetReferenceCount()const;//被引用次数
		protected:
			Ref();//初始化
		public:
			virtual~Ref();//析构
		protected:
			unsignedint_referenceCount;//引用次数
			friendclassAutoreleasePool;//自动释放池
	};


/**
*	CCRef.cpp
**/
	//节点被创建时,引用次数为1
	Ref::Ref():_referenceCount(1)
	{
	}

	voidRef::retain()
	{
		CCASSERT(_referenceCount>0,"referencecountshouldgreaterthan0");
		++_referenceCount;
	}

	voidRef::release()
	{
		CCASSERT(_referenceCount>0,"referencecountshouldgreaterthan0");
		--_referenceCount;
		if(_referenceCount==0)
		{
			deletethis;
		}
	}

	Ref*Ref::autorelease()
	{
		//将节点加入自动释放池
		PoolManager::getInstance()->getCurrentPool()->addObject(this);
		returnthis;
	}
//

Ref原理分析:

>当一个 Ref 初始化(被new出来时),_referenceCount = 1;

>当调用该 Ref 的 retain() 方法时,_referenceCount++;

>当调用该 Ref 的 release() 方法时,_referenceCount--。

>若_referenceCount 减后为0,则 delete 该 Ref。


2、retain() 和 release() 使用

下面一段简单的例子来学习 retain() 和 release() 的使用。

//
	TestObject*obj1=newTestObject("testobj1");
	CCLOG("obj1referenceCount=%d",obj1->getReferenceCount());
	
	obj1->retain();
	CCLOG("obj1referenceCount=%d",obj1->getReferenceCount());
	
	obj1->release();
	CCLOG("obj1referenceCount=%d",obj1->getReferenceCount());
	
	obj1->release();
//

控制台显示的日志如下:

cocos2d: TestObject:testobj1 is created

cocos2d: obj1 referenceCount=1

cocos2d: obj1 referenceCount=2

cocos2d: obj1 referenceCount=1

cocos2d: TestObject:testobj1 is destroyed


通过例子和打印结果可以看到:

>obj1对象创建后,引用计数为1;

>执行一次retain()后,引用计数为2;

>执行一次release()后,引用计数回到1;

>再执行一次release()后,对象会被释放掉。

因此:

>我们可以调用retain()方法,令其引用计数增1,表示获取该对象的引用权;

>在引用结束的时候调用release()方法,令其引用计数值减1,表示释放该对象的引用权。

>直到对象的引用计数为0,释放该对象。


3、autorelease() 使用

同样一段简单的例子来学习autorelease的使用,代码如下:

//
	TestObject*obj=newTestObject("testobj");
	CCLOG("objreferenceCount=%d",obj->getReferenceCount());
	
	obj->autorelease();
	CCLOG("objisaddincurrentpool%s",PoolManager::getInstance()->getCurrentPool()->contains(obj)?"true":"false");
	CCLOG("objreferenceCount=%d",obj->getReferenceCount());
	
	obj->retain();
	CCLOG("objreferenceCount=%d",obj->getReferenceCount());

	obj->release();
	CCLOG("objreferenceCount=%d",obj->getReferenceCount());

	//objincurrentpoolwillberelease
	Director::getInstance()->replaceScene(this);
//

cocos2d: TestObject:testobj is created

cocos2d: obj referenceCount=1

cocos2d: obj is add in currentpool true

cocos2d: obj referenceCount=1

cocos2d: obj referenceCount=2

cocos2d: obj referenceCount=1

...

cocos2d: TestObject:testobj is destroyed


通过代码和打印结果,我们可以看到:

>obj对象创建后,引用计数为1;

>执行一次autorelease()后,obj对象被加入到当前的自动释放池。

>obj对象的引用计数值并没有减1。

>但是在下一帧开始前,当前的自动释放池会被回收掉,并对自动释放池中的所有对象执行一次release()操作。

>当对象的引用计数为0时,对象会被释放掉。


>obj对象执行autorelease()后,我们对其执行了一组retain()和release()操作。

>此时obj对象的引用计数为1,在场景切换后,当前的自动释放池被回收,

>obj对象执行一次release()操作引用计数减为0时,对象会被释放掉。


注意:autorelease()只有在自动释放池被释放时才会进行一次释放操作,如果对象释放的次数超过了应有的次数,则这个错误在调用autorelease()时并不会被发现,只有当自动释放池被释放时(通常也就是游戏的每一帧结束时),游戏才会崩溃。在这种情况下,定位错误就变得十分困难了。

例如,在游戏中,一个对象含有1个引用计数,但是却被调用了两次autorelease()。在第二次调用autorelease()时,游戏会继续执行这一帧,结束游戏时才会崩溃,很难及时找到出错的地点。

因此,我们建议在开发过程中应该避免滥用autorelease(),只在工厂方法等不得不用的情况下使用,尽量以release()来释放对象引用。


4、AutoreleasePool类 使用

Cocos2d-x提供AutoreleasePool,管理自动释放对象

下面一段简单的例子讲解AutoreleasePool的使用,代码如下:

//
	TestObject*obj2=newTestObject("testobj2");
	CCLOG("obj2referenceCount=%d",obj2->getReferenceCount());
	
	//useAutoreleasePool
	{
		AutoreleasePoolpool;
	
		obj2->retain();
		CCLOG("obj2referenceCount=%d",obj2->getReferenceCount());
	
		obj2->release();
		CCLOG("obj2referenceCount=%d",obj2->getReferenceCount());
	
		obj2->autorelease();
		CCLOG("obj2isaddinpool%s",pool.contains(obj2)?"true":"false");
	
		TestObject*obj3=newTestObject("testobj3");
		
		obj3->autorelease();
		CCLOG("obj3isaddinpool%s",pool.contains(obj3)?"true":"false");
	}
//

控制台输出日志如下:

cocos2d: TestObject:testobj2 is created

cocos2d: obj2 referenceCount=1

cocos2d: obj2 referenceCount=2

cocos2d: obj2 referenceCount=1

cocos2d: obj2 is add in pool true

cocos2d: TestObject:testobj3 is created

cocos2d: obj3 is add in pool true

cocos2d: TestObject:testobj2 is destroyed

cocos2d: TestObject:testobj3 is destroyed


通过代码和输出结果,可以看到:

>创建了一个obj2对象,此时obj2对象的引用计数为1。

>接着创建了一个自动释放池,对obj2对象执行retain()和release()操作后,执行autorelease()操作,此时obj2对象被加入到当前新建的自动释放池中。

>接着新建了obj3对象,并执行autorelease()操作。同样obj3也被加入到当前新建的自动释放池中。

>在代码块结束后,自动释放池被回收,加入自动释放池中的obj2和obj3执行release()操作,引用计数减为0,被释放销毁。


我们可以自己创建AutoreleasePool,管理对象的autorelease。

我们已经知道,调用了autorelease()方法的对象(下面简称"autorelease对象"),将会在自动释放池释放的时候被释放一次。虽然,Cocos2d-x已经保证每一帧结束后释放一次释放池,并在下一帧开始前创建一个新的释放池,但是我们也应该考虑到释放池本身维护着一个将要执行释放操作的对象列表,如果在一帧之内生成了大量的autorelease对象,将会导致释放池性能下降。因此,在生成autorelease对象密集的区域(通常是循环中)的前后,我们最好可以手动创建并释放一个回收池。

例如:

//
	//exampleofusingtempleautoreleasepool
	{
		AutoreleasePoolpool2;
		charname[20];
		for(inti=0;i<100;++i)
		{
			snprintf(name,20,"object%d",i);
			TestObject*tmpObj=newTestObject(name);
			tmpObj->autorelease();
		}
	}
//

总结:

>autorelease()的实质是将对象加入自动释放池,对象的引用计数不会立刻减1,在自动释放池被回收时对象执行release()。

> autorelease()并不是毫无代价的,其背后的释放池机制同样需要占用内存和CPU资源。

>过多的使用autorelease()会增加自动释放池的管理和释放池维护对象存取释放的支出。

>在内存和CPU资源本就不足的程序中使得系统资源更加紧张。

>此时就需要我们合理创建自动释放池管理对象autorelease。

>不用的对象推荐使用release()来释放对象引用,立即回收。


5、特殊内存管理


5.1、工厂方法 create()

在Cocos2d-x中,提供了大量的工厂方法创建对象。仔细看你会发现,这些对象都是自动释放的。

下面以 Label 的 create 方法为例,代码如下:

//
	Label*Label::create()
	{
		autoret=newLabel();
		if(ret)
		{
			ret->autorelease();
		}
		returnret;
	}
//

我们可以发现,创建了一个Label的对象,并对该对象执行autorelease()。表示该对象是自动释放的。细心的你会发放Layer/Scene/Sprite等类的 create() 方法都相同。

使用工厂方法创建对象时,虽然引用计数也为1,但是由于对象已经被放入了释放池,因此调用者没有对该对象的引用权,除非我们人为地调用了retain()来获取引用权,否则,不需要主动释放对象。


5.2、Node 的 addChild() / removeChild 方法

在Cocos2d-x中,所有继承自Node类,在调用 addChild 方法添加子节点时,自动调用了retain。 对应的通过removeChild,移除子节点时,自动调用了release。

调用addChild方法添加子节点,节点对象执行retain。子节点被加入到节点容器中,父节点销毁时,会销毁节点容器释放子节点。对子节点执行release。如果想提前移除子节点我们可以调用removeChild。

在Cocos2d-x内存管理中,大部分情况下我们通过调用 addChild/removeChild 的方式自动完成了retain,release调用。不需再调用retain,release。



【内存优化】


1、内存优化原理

为优化应用内存使用,开发人员首先应该知什么最耗应用内存,答案就是纹理! 纹理几乎会占据90%应用内存。所以尽量最小化应用的纹理内存使用,否则应用很有可能会因为低内存而崩溃。

本节介绍Cocos2d-x游戏通用的两条内存优化原理指导


1.1、认识瓶颈寻找方案

什么样的纹理最耗应用内存?或这些纹理会消耗多少内存?

当然这个不用手动计算,只需猜测。工具在这里已经准备好了,使用的是苹果的工具“Allocation & Leaks”。你可以在Xcode中长按“Run”命令,选择“ Profile ”来启动这两个工具。

如下所示:

wKiom1TUx5TC2FpgAAD0F4fYCY4427.jpg


使用Allocation工具可以监控应用的内存使用,使用Leaks工具可以观察内存的泄漏情况。

此外还可用一些代码获取游戏内存使用的其他信息。

如下所示:

//
	Sprite*bg=Sprite::create("HelloWorld.png");
	bg->setPosition(240,160);
	this->addChild(bg);

	CCLOG("%s",Director::getInstance()->getTextureCache()->getCachedTextureInfo().c_str());
//

调用这个代码后,游戏便会在DEBUG模式运行,这时你会在Xcode控制台窗口看到一些格式工整的日志信息。

//
	cocos2d:"****/HelloWorld.png"rc=2id=3480x320@32bpp=>600KB
	"/cc_fps_images"rc=5id=2999x54@16bpp=>105KB
	TextureCachedumpDebugInfo:2textures,for705KB(0.69MB)
//

从上可以看到会显示纹理的名称、引用计数、ID、大小及每像素的位数。最重要的是会显示内存的使用情况。如“cc_fps_images”指消耗了105KB内存,而“HelloWorld.png”消耗了600KB内存。


1.2、切勿过度优化

这是一个通用的优化规则。在优化过程中,应该做一些权衡取舍。因为有时候图像质量和图像内存使用是处于两级的状态。千万不要过度优化!


2、内存优化水平

在此将ccos2d-x内存优化分为:三个等级

每个等级都有不同的说明,策略也有点不一样。


2.1、客户端等级

这是最重要的的优化等级。因为我们要在Cocos2d-x引擎顶层编译游戏,引擎自身会提供一些优化选项。 在这个等级我们可以进行大部分优化。简而言之,我们可以优化纹理、音频、字体及粒子的内存使用。

第一: 看纹理优化,为了优化纹理内存使用,必须知道什么因素对纹理内存使用的影响最大。主要有3个因素会影响纹理内存,即纹理格式(压缩还是非压缩)、颜色深度和大小。我们可以使用PVR格式纹理减少内存使用。推荐纹理格式为pvr.ccz。纹理使用的每种颜色位数越多,图像质量越好,但是越耗内存。所以我们可以使用颜色深度为RGB4444的纹理代替RGB8888,这样内存消耗会降低一半。此外超大的纹理也会导致内存相关问题。所以最好使用中等大小的纹理。

第二: 音频优化,3个因素会影响音频文件的内存使用,即音频文件数据格式、比特率及采样率。推荐使用MP3数据格式的音频文件,因为Android平台和iOS平台均支持MP3格式,此外MP3格式经过压缩和硬件加速。背景音乐文件大小应该低于800KB,最简单的方法就是减少背景音乐时间然后重复播放。音频文件采样率大约在96-128kbps为佳,比特率44kHz就够了。

第三:字体和粒子优化,在此有两条小提示:使用BMFont字体显示游戏分数时,请尽可能使用最少数量的文字。例如只想要显示单位数的数字,你可以移除所有字母。至于粒子,可以通过减少粒子数来降低内存使用。


2.2、引擎等级

需要 OpenGL ES 及游戏引擎高手。


2.3、C++语言等级

在这个等级中,建议是编写无内存泄露代码。遵循Cocos2d-x内置的内存管理原则,尽量避免内存泄露。


3.、提示和技巧

(1) 一帧一帧载入游戏资源

(2) 减少绘制调用,使用“Auto-batching”自动批处理。

(3) 载入纹理时按照从大到小的顺序

(4) 避免高峰内存使用

(5) 使用载入屏幕预载入游戏资源

(6) 需要时释放空闲资源

(7) 收到内存警告后释放缓存资源.

(8) 使用纹理打包器优化纹理大小、格式、颜色深度等

(9) 使用JPG格式要谨慎!

(10) 请使用RGB4444颜色深度16位纹理

(11) 请使用NPOT纹理,不要使用POT纹理

(12) 避免载入超大纹理

(13) 推荐1024*1024 NPOT pvr.ccz纹理集,而不要采用RAW PNG纹理

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


    本文实践自 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在载入图片方面也有了非常大改变,仅仅只是