Cocos2dx对精灵的优化
cocos2dx针对游戏设计的不同方面会有不同的优化方案,可以对声音,对内存,对图片格式,对色彩等等进行优化。有关这些方面的方法请大家查找其他的文章。我今天要说的是如何对精灵进行优化,程序中我们用到的最多的就是精灵,大到背景、UI,小到 NPC、道具,只要是用图片展示的,都是精灵或它的子类。精灵是什么,在我看来精灵就是一张纹理图片,是按某种方式显示出来的图片。精灵如此的重要,我们当然要好好的优化优化了。我们可以减小精灵图片的大小,使用缓存Cache的方法将精灵提前加载到内存中,当有很多精灵的时候,使用批次渲染的方法。所以我就从缓存和批次渲染这俩个方面来说一下如何优化我们的精灵图片。
1、通过批次渲染的方法来优化精灵。在游戏中的某一时刻,有时候会用到大量的精灵,比如说发射子弹,粒子效果,这些精灵图片所使用的纹理都是相同的,如果一张一张的图片进行渲染势必会降低效率,大家在开发中看到的窗口的左下角的三行数字中,第一行数字就是渲染批次,这个渲染批次在我看来就是画了多少次,cocos2dx中使用opengl进行绘图,渲染的批次越少当然越好了,所以这个渲染批次才会显示在左下角让我们参考。我们应当尽量的减小这个渲染的批次,提高游戏的效率。方法就是使用CCSpriteBatchNode和CCParticleBatchNode精灵批节点类和粒子批节点类。先来看看在代码中如何使用它们。
<pre class="brush:cpp; toolbar: true; auto-links: false;">
CCSize visibleSize = CCDirector::sharedDirector()->getVisibleSize();
//创建了十个精灵将这十个精灵添加到当前的层中
for(int i=0;i<10;i++)
{
CCSprite * sprite = CCSprite::create("icon.png");
sprite->setPosition(ccp(CCRANDOM_0_1()*visibleSize.width,CCRANDOM_0_1()*visibleSize.height));
this->addChild(sprite);
}
</pre>
<pre class="brush:cpp; toolbar: true; auto-links: false;">
//创建一个CCSpriteBatchNode,传入的参数就是精灵们将要用到的图片
CCSpriteBatchNode * batchNode = CCSpriteBatchNode::create("icon.png");
//或者使用texture2d初始化,里边传入一个texture
//CCSpriteBatchNode * batch = CCSpriteBatchNode::createWithTexture();
//这一句写不写都可以,因为node的默认坐标就是(0,0)
batchNode->setPosition(CCPointZero);
for(int i=0;i<10;i++)
{
//创建的这些精灵所使用的纹理必须和CCSpriteBatchNode相同,而且所有这些精灵必须在同一个渲染层
CCSprite * sprite = CCSprite::createWithTexture(batchNode->getTexture());
sprite->setPosition(ccp(CCRANDOM_0_1()*visibleSize.width,CCRANDOM_0_1()*visibleSize.height));
batchNode->addChild(sprite);
}
this->addChild(batchNode);
</pre>

<pre class="brush:cpp; toolbar: true; auto-links: false;">
CCTexture2D * texture = CCTextureCache::sharedTextureCache()->addImage("fire.png");
for(int i=0;i<10;i++)
{
//CCParticleSun里边没有参数
CCParticleSystem * particle = CCParticleSun::create();
particle->setTexture(texture);
particle->setPosition(ccp(CCRANDOM_0_1()*visibleSize.width,CCRANDOM_0_1()*visibleSize.height));
this->addChild(particle);
}
</pre>

<pre class="brush:cpp; toolbar: true; auto-links: false;">
CCTexture2D * texture = CCTextureCache::sharedTextureCache()->addImage("fire.png");
//参数同样需要一张纹理图片
CCParticleBatchNode * particleBatch = CCParticleBatchNode::createWithTexture(texture);
//也可以采用如下的方式创建
//CCParticleBatchNode * particleBatch = CCParticleBatchNode::create("fire.png");
for(int i=0;i<10;i++)
{
//CCParticleSun里边没有参数
CCParticleSystem * particle = CCParticleSun::create();
//传入的texture必须和CCParticleBatchNode相同
particle->setTexture(texture);
particle->setPosition(ccp(CCRANDOM_0_1()*visibleSize.width,CCRANDOM_0_1()*visibleSize.height));
particleBatch->addChild(particle);
}
this->addChild(particleBatch);
</pre>

2、使用缓存提前加载精灵。当我们使用纹理的时候可以制作一个loading界面,将将要用到的纹理加载到缓存中,同时将它们的引用计数增加一,以便这些纹理不会被释放。用的时候直接从缓存中取就可以了,这样也可以提高效率。我们用到的缓存类有CCTextureCache、CCSpriteFrameCache、CCAnimationCache,下面分别说明其用法。
<pre class="brush:cpp; toolbar: true; auto-links: false;">
bool HelloWorld::init()
{
//////////////////////////////
// 1. super init first
if ( !CCLayer::init() )
{
return false;
}
CCSize visibleSize = CCDirector::sharedDirector()->getVisibleSize();
//缓存其实就是一个数组,把用到的纹理图片放到这个数组中,纹理的引用计数加1,这样的话就不会释放纹理图片了
//等下一次使用的时候直接从这个数组中取就可以了,这样的话就不必再次加载到内存中了
CCTexture2D * texture = CCTextureCache::sharedTextureCache()->addImage("icon.png");
CCLog("%d",texture->retainCount()); //count=1
//使用如下俩种方法可以获得缓存中的纹理,第二种方法如果之前已经加载了纹理,这个时候不会重新加载
//而是直接返回
texture = CCTextureCache::sharedTextureCache()->addImage("icon.png");
texture = CCTextureCache::sharedTextureCache()->textureForKey("icon.png");
CCLog("%d",texture->retainCount()); //count=1
//异步加载图片,开辟一个新的线程专门用来加载图片,加载完毕以后调用loadingCallBack函数
//所以在这个init函数中是不应该去调用函数textureForKey的,因为你不知道是否加载好了纹理啊
CCTextureCache::sharedTextureCache()->addImageAsync("icon1.png",
this,callfuncO_selector(HelloWorld::loadingCallBack));
return true;
}
//object就是加载好了的纹理
void HelloWorld::loadingCallBack(CCObject * object)
{
CCTexture2D * texture = (CCTexture2D *)object;
CCLog("%d",texture->retainCount()); //count=2
//通过以下的方法取得的texture和上边的那个texture相同
//CCTexture2D * texture2 = CCTextureCache::sharedTextureCache()->textureForKey("icon1.png");
//CCLog("%d",texture2->retainCount());
//会清除掉所有的纹理,在其他的地方不可以再引用这些纹理了
CCTextureCache::sharedTextureCache()->removeAllTextures();
//以下的方法会remove掉count值为1的纹理,icon.png被remove掉了,个人认为实际用的时候就用这个
//CCTextureCache::sharedTextureCache()->removeUnusedTextures();
//count=1
CCLog("%d",texture->retainCount());
//查看纹理清除的信息
CCTextureCache::sharedTextureCache()->dumpCachedTextureInfo();
//切换场景
CCDirector::sharedDirector()->replaceScene(TestScene::scene());
}
</pre>
在新的场景中使用加载好的纹理。
<pre class="brush:cpp; toolbar: true; auto-links: false;">
bool TestScene::init()
{
//因为将所有的纹理清除掉了,所以这里引用的时候会出错
CCTexture2D * texture = CCTextureCache::sharedTextureCache()->textureForKey("icon1.png");
CCLog("%d",texture->retainCount());
return true;
}
</pre>
CCTextureCache在我看来就是一个数组,将加载的纹理都放到这个数组中,然后将它们的引用计数加1以防释放掉,我们要使用加载好的纹理直接从这个数组中取就可以了,返回一个CCTexture2D的对象,然后我们使用CCSprite:createWithTexture这个方法来创建出精灵。我觉的这里最主要的问题就是纹理的释放,个人认为使用removeUnusedTexture这个函数比较好,它会释放掉引用计数为1的纹理,也就说明程序中没有再为它增加引用计数,肯定是没用了。那些大于1的纹理不会释放掉,即使我们在不知道的情况下再次加载,也只是返回已经加载好了的。接下来是另外俩个缓存了,和这个意思差不多,直接看代码吧。
<pre class="brush:cpp; toolbar: true; auto-links: false;">
bool HelloWorld::init()
{
//////////////////////////////
// 1. super init first
if ( !CCLayer::init() )
{
return false;
}
CCSize visibleSize = CCDirector::sharedDirector()->getVisibleSize();
//可以使用texturepacker生成.plist文件,将plist文件和png文件放到资源目录下,通过add方法将ghosts.plist纹理
//加载到了缓存中
CCSpriteFrameCache::sharedSpriteFrameCache()->addSpriteFramesWithFile("ghosts.plist");
//spriteFrameByName中传入的参数可以到.plist文件中查看,通过这个方法可以从缓存中获得一个精灵帧
CCSpriteFrame * spriteFrame = CCSpriteFrameCache::sharedSpriteFrameCache()->spriteFrameByName("child1.gif");
CCLog("%d",spriteFrame->retainCount()); //count=1
//之所以创建精灵帧就是为了播放动画做准备的,这里可以是一个循环不到的创建精灵帧然后添加到animation中
//关于这部分的内容可以查看我前边的文章,这里是为了说明缓存的使用
CCAnimation * animation = CCAnimation::create();
animation->addSpriteFrame(spriteFrame);
CCLog("%d",spriteFrame->retainCount()); //count=2
CCLog("%d",animation->retainCount()); //count=1,这个1是创建时候的引用计数
//动画缓存,将animation放到缓存中,里边存入一个key,方便以后取出,如果不把动画放到这个缓存中,下一帧的时候
//动画就会被释放掉的,再播放动画就会出错了
CCAnimationCache::sharedAnimationCache()->addAnimation(animation,"a"); //count=2
CCLog("%d",animation->retainCount());
CCAnimate * animate = CCAnimate::create(CCAnimationCache::sharedAnimationCache()->animationByName("a"));
//当不再需要播放动画的时候从动画缓存中清除,应该先清除动画再清除精灵帧
CCAnimationCache::sharedAnimationCache()->removeAnimationByName("a");
//将引用计数为1的精灵帧从缓存中清除
CCSpriteFrameCache::sharedSpriteFrameCache()->removeUnusedSpriteFrames();
//释放掉所有的精灵帧
//CCSpriteFrameCache::sharedSpriteFrameCache()->removeSpriteFrames();
return true;
}
</pre>
原文转载:http://www.zaojiahua.com/optimization.html皂荚花。
Cocos2dx对精灵的优化的更多相关文章
- 0428数字口袋精灵app优化
"数字口袋精灵app"优化 目录: 一.项目github总仓库推送 二.开发成员 三.分工与合作 四.各模块成果 五.团队成员贡献分 内容: 一.项目github总仓库: http ...
- cocos2dx SpriteBatchNode 精灵的渲染优化类
用处是减少对精灵的渲染次数,用法如下 SpriteFrameCache::getInstance()->addSpriteFramesWithFile("person.plist&qu ...
- cocos2d-x 让精灵按照自己设定的运动轨迹行动
转自:http://blog.csdn.net/ufolr/article/details/7447773 在cocos2d中,系统提供了CCMove.CCJump.CCBezier(贝塞尔曲线)等让 ...
- [COCOS2DX]交叉编译实践+速度优化(vs2012修改win32代码+修改makefile+编译安卓项目包+部署安卓项目包到Eclipse+运行apk)
通过前面的部署过程可以知道cocos2dx的开发过程如下: 1.VS2012完成修改 2.因为指定了CPP文件位置,ndk可以通过jni方式完成C++文件的编译,运行以下命令完成proj.androi ...
- [cocos2d-x] 让精灵响应触摸 并把方向旋转到相对应的角度
在cocos2d-x里面 想要把一个精灵从原位置移动到用户所触摸到的点 , 并且把精灵的方向旋转相对应的弧度,可以参考一下我的做法 我这里的精灵是用一条鱼, 用户触摸后鱼就移动到所触摸的点, 并且移 ...
- cocos2dx 获取精灵的高亮效果
转自:http://blog.csdn.net/tyxkzzf/article/details/38703883 CCSprite* getHighlightSprite(CCSprite* norm ...
- [windows+cocos2dx]CCSprite精灵类
序言 回想cocos2dx,之前在mac+Xcode平台学习了一遍cocos2dx,一年时间不接触cocos了.一直在搞Unity3d.如今还是就之前所学温故温故,但不再用Xcode来写.用经常使用的 ...
- cocos2d-x创建精灵动画方式汇总
1.创建精灵框架缓存,并向其中添加相应的动画文件(plist),最后,通过动画集缓存生产动画 CCSpriteFrameCache *cache = CCSpriteFrameCache::share ...
- Cocos2d-x 让精灵随手指移动起来二(简单实现)
void HelloWorld::ccTouchMoved(cocos2d::CCTouch *touch, cocos2d::CCEvent *event) { CCSize winSize = C ...
随机推荐
- github使用
1.首先登录到https://github.com注册Github帐号,并且创建一个repository. 例如:注册的github帐号名为whu-zhangmin,创建的repository名称为w ...
- Excel函数汇总:
/** *D1—要查找的目标值 *G:G—查找的单元格范围,G:G表示G列 *1—查找第一个匹配 *FALSE—找到结果即返回 */ VLOOKUP(D1,G:G,1,FALSE):返回查找到的单元格 ...
- java向Excel文件写入数据
/*使用之前要记得导入第三的jar包这个是我之前使用的时候那别人的东西自己修改了一下 还没来得及好好地封装一下还望见谅,注释我感觉写的挺清楚的就在不进行解释代码了*/package com.zzp.E ...
- 4412开发板升级4.2之后改了logo开机后屏幕闪解决办法
荣品4412开发板升级到4.2请注意增加虚拟机内存. 问:荣品4412开发板升级到Android4.2之后,改了logo.4412板子开机后,过一会屏幕就一闪一闪,是什么原因? Android4.2编 ...
- MINA2 框架详解(转)
Apache Mina Server 是一个网络通信应用框架,也就是说,它主要是对基于TCP/IP.UDP/IP协议栈的通信框架(当然,也可以提供JAVA 对象的序列化服务.虚拟机管道通信服务等),M ...
- 1171. Lost in Space
http://acm.timus.ru/problem.aspx?space=1&num=1171 一天的时间,WA了N遍,居然是因为数组开小了呀,我勒个去!鄙视自己...... 我是从第 1 ...
- 《JavaScript高级程序设计》读书笔记--(3)引用类型
ECMAScript从技术来说是一门面向对象的语言,但不具备传统的面向对象语言所支持的类和接口等基本结构.虽然引用类型与类看起来类似,但是他们并不是相同的概念.引用类型有时也被成为对象定义,因为它描述 ...
- ASP.NET中Onclick和OnserverClick事件的区别
对于服务器按钮控件(即<asp:Button>类型的按钮): 服务器响应事件:OnClick 客户端响应属性:OnClientClick 对于html按钮控件(即<input typ ...
- Cheap Hollister Clothing
(link to hollisterco site), Spectacles don't simply take care of the eye area inside sun; Putting th ...
- JS技术大全
事件源对象:event.srcElement.tagName event.srcElement.type 捕获/释放:event.srcElement.setCapture(); event.sr ...