http://blog.csdn.net/w20175357/article/details/23546985

1、先说说addImageAsync()异步加载图片的问题

做游戏的时候现在资源的比较大,所有我们必然会有一个loading界面,而我在找写loading界面的方法的时候,发现了2种方法。

      一种是自己创建一个线程,再在这个线程里面加载资源,不过由于openGL的限制,只能在主线程里面绘制UI,不过有的人也想出了其他的方法,就是先只缓存,再在主线程里面绘制。这里有这方面的教程。不过这种方法有点绕,反正我是搞得不是很清楚。

    另一种,也是TestCpp里面使用的方法,那就是用

CCTextureCache::sharedTextureCache()->addImageAsync("Texture2D_Test3.png", this, callfuncO_selector(LoadingScene::loadingCallBack));

函数异步加载图片,基本应用可以看TestCpp里面的TextureCacheTest这一个。

2、我遇到的问题

TestCpp里面当资源都加载好了的时候,它并没有跳转场景,而是把先前的画面显示的都移除,再添加新的精灵,但是我现在想做是当资源都加载好的时候就从loading界面跳转到读取成功界面,而不是简单的把资源移除与添加新的资源。TextureCacheTest里面的写法如下。详细的请自行去TestCpp里面查看。

if (m_nNumberOfLoadedSprites == m_nNumberOfSprites)
{
this->removeChild(m_pLabelLoading, true);
this->removeChild(m_pLabelPercent, true);
addSprite(); //仅仅添加精灵
}

但是我想做的是

if (m_nLoadedResources == m_nAllLoadResources)
{
this->removeAllChildren();
CCScene *loginSuccessScene = LoginSuccessScene::scene();
CCDirector::sharedDirector()->replaceScene(CCTransitionFade::create(0.1f, loginSuccessScene)); }

        不过当我这么写的时候就遇到了问题,那就是场景确实是跳转了,但是只是一闪而逝,就又跳回到了loading的场景里面,而由于我this->removeAllChildren(),所以最后屏幕就黑了。

3、我的解决方法

关于上面那个问题,我的理解是由于我是在addImageAsync的回调函数LoadingScene::loadingCallBack 里面写的跳转场景,但是当这个线程执行完了之后,主线程会还是在执行的。所以这个里面的跳转只是一闪而逝的。然后我请教他人知道了其实主线程里面有一个update()函数,它会一直调用。于是我便把上面的那个函数写在了update里面,因为update是引擎自带的函数,所以虽然我显性没有调用,但是它会自行调用,你可以直接查看update()函数的原型。update()如下

void LoadingScene::update( float delta )
{
if (m_nLoadedResources == m_nAllLoadResources)
{
this->removeAllChildren();
CCScene *loginSuccessScene = LoginSuccessScene::scene();
CCDirector::sharedDirector()->replaceScene(CCTransitionFade::create(0.1f, loginSuccessScene));
}
}

但是这样其实会有一个问题,当资源读取很快的时候,会跳到loading成功界面,不过之后还是会闪一下,我的理解是当跳转的时候,可能异步加载资源海没有完全完成,所以还是会运行那个函数。所以我便加了一个变量m_fWaitTime,在update里面让其m_fWaitTime += delta;   并当m_fWaitTime > 5.0f的时候才跳转页面,当你想测试一个不等待的时候的效果的时候可以将这个变量相关的东西都注释掉,当然了,我们运行游戏的时候加载时间一般都相对较长,所以不用担心。(不过我还没有真机测试过=.=,如果真机测试的时候出问题了,我会过来及时更正的)。

4、loading源代码

下面是loading的主要代码,其中addImageAsync()加载的资源都可以用CCTexture2D *texture1 = CCTextureCache::sharedTextureCache()->textureForKey("Test.png");直接取的并使用,就算是在不同的类里面使用而是同一个类。

LoadingScene.h

#ifndef _LOADING_SCENE_H__
#define _LOADING_SCENE_H__ #include "cocos2d.h" USING_NS_CC; class LoadingScene : public CCLayer
{
public:
virtual bool init(); static CCScene *scene(); void loadSuccess(); //读取的回调函数
void loadingCallBack(CCObject *obj); void addSprite();
void update(float delta);
void updateProgress(float dt); CREATE_FUNC(LoadingScene); private:
//读取开始时候的进度条
CCSprite *m_pLoadBarStart; //读取完成时候的进度条
CCProgressTimer *m_pLoadBarEnd; //线程相关的函数
//总的加载图片数
int m_nAllLoadResources; //当前加载图片数
int m_nLoadedResources; //读取进度
float m_fProgressIndex; CCLabelTTF *m_pLabelLoading;
CCLabelTTF *m_pLabelPercent; // 当要测试有m_fWaitTime的效果的时候,将下一行与update()和init()里面初始化的注释取消掉即可
// float m_fWaitTime; };
#endif

LoadingScene.cpp

#include "LoadingScene.h"
#include "LoginSuccessScene.h"
#include "Global.h" CCScene* LoadingScene::scene()
{
CCScene * scene = NULL;
do
{
// 'scene' is an autorelease object
scene = CCScene::create();
CC_BREAK_IF(! scene); // 'layer' is an autorelease object
LoadingScene *layer = LoadingScene::create();
CC_BREAK_IF(! layer); // add layer as a child to scene
scene->addChild(layer);
} while (0); // return the scene
return scene;
} bool LoadingScene::init()
{
if (!CCLayer::init())
{
return false;
}
scheduleUpdate();
CCSize size = CCDirector::sharedDirector()->getWinSize(); m_nAllLoadResources = 3;
m_nLoadedResources = 0;
m_fProgressIndex = 0.0;
// m_fWaitTime = 0; m_pLabelLoading = CCLabelTTF::create("loading...", "Arial", 15);
m_pLabelPercent = CCLabelTTF::create("%0", "Arial", 15); m_pLabelLoading->setPosition(ccp(400, 200));
m_pLabelPercent->setPosition(ccp(450, 200)); this->addChild(m_pLabelLoading, 1);
this->addChild(m_pLabelPercent, 1); //loading的动画效果
m_pLoadBarStart = CCSprite::create("loadingStart.jpg");
m_pLoadBarStart->setPosition(ccp(size.width / 2, size.height * 3 / 4));
m_pLoadBarStart->setScale(2.0f);
this->addChild(m_pLoadBarStart); m_pLoadBarEnd = CCProgressTimer::create(CCSprite::create("loadingEnd.jpg"));
m_pLoadBarEnd->setPercentage(1.0f);
m_pLoadBarEnd->setPosition(ccp(size.width / 2, size.height * 3 / 4));
m_pLoadBarEnd->setType(kCCProgressTimerTypeBar);
m_pLoadBarEnd->setBarChangeRate(ccp(1, 0));
m_pLoadBarEnd->setMidpoint(ccp(0, 0));
m_pLoadBarEnd->setScale(2.0f);
this->addChild(m_pLoadBarEnd); //读取资源
CCTextureCache::sharedTextureCache()->addImageAsync("Texture2D_Test1.png", this, callfuncO_selector(LoadingScene::loadingCallBack)); CCTextureCache::sharedTextureCache()->addImageAsync("Texture2D_Test2.png", this, callfuncO_selector(LoadingScene::loadingCallBack));
CCTextureCache::sharedTextureCache()->addImageAsync("Texture2D_Test3.png", this, callfuncO_selector(LoadingScene::loadingCallBack)); //设置一个动作,令进度条2秒内读取到100%
// CCProgressTo *action = CCProgressTo::create(2, 100);
// m_pLoadBarEnd->runAction(CCSequence::create(action, CCCallFunc::create(this,
// callfunc_selector(LoadingScene::loadSuccess)), NULL)); return true;
} void LoadingScene::loadSuccess()
{
this->removeAllChildren();
CCScene *loginSuccessScene = LoginSuccessScene::scene();
CCDirector::sharedDirector()->replaceScene(CCTransitionFade::create(0.1f, loginSuccessScene));
// CCDirector::sharedDirector()->pushScene(loginSuccessScene); } void LoadingScene::loadingCallBack(CCObject *obj)
{
++m_nLoadedResources;
char tmp[10];
sprintf(tmp,"%%%d", (int)(((float)m_nLoadedResources / m_nAllLoadResources ) * 100));
m_pLabelPercent->setString(tmp); m_fProgressIndex = (((float)m_nLoadedResources / m_nAllLoadResources ) * 100);
m_pLoadBarEnd->setPercentage(m_fProgressIndex); } void LoadingScene::update( float delta )
{
// m_fWaitTime += delta;
if (m_nLoadedResources == m_nAllLoadResources
// && m_fWaitTime > 5.0f
)
{
loadSuccess();
}
}

如果有什么错误或者解释不当的地方欢迎指正,我只是因为这个困扰了我的问题得到了解决便写上来,让也被这个问题困扰的人能够得到解答。希望能够帮到你们。

cocos2d-x addImageAsync()异步加载资源成功之后的场景跳转问题的更多相关文章

  1. 【Cocos2d-Js基础教学(5)资源打包工具的使用及资源的异步加载处理】

    TexturePacker是纹理资源打包工具,支持Cocos2dx的游戏资源打包. 如果用过的同学可以直接看下面的资源的异步加载处理 首先为什么用TexturePacker? 1,节省图片资源实际大小 ...

  2. cocos2d-x lua中实现异步加载纹理

    原文地址:  http://www.cnblogs.com/linchaolong/p/4033118.html 前言   问题:最近项目中需要做一个loading个界面,界面中间有一个角色人物走动的 ...

  3. Unity跳转场景进度条制作教程(异步加载)

    Unity跳转场景进度条制作 本文提供全流程,中文翻译. Chinar 坚持将简单的生活方式,带给世人!(拥有更好的阅读体验 -- 高分辨率用户请根据需求调整网页缩放比例) Chinar -- 心分享 ...

  4. AssetBundle异步加载被中断的问题

    刘 刘泰言创建于 1 年前 在使用异步接口 yield return AssetBundle.ASyncLoad的时候,难免会想到:这个异步处理完之前如何Cancel掉这个任务?也就是一个AssetB ...

  5. Aery的UE4 C++游戏开发之旅(4)加载资源&创建对象

    目录 资源的硬引用 硬指针 FObjectFinder<T> / FClassFinder<T> 资源的软引用 FSoftObjectPaths.FStringAssetRef ...

  6. [Android] Android 用于异步加载 ContentProvider 中的内容的机制 -- Loader 机制 (LoaderManager + CursorLoader + LoaderManager.LoaderCallbacks)

    Android 用于异步加载 ContentProvider 中的内容的机制 -- Loader 机制 (LoaderManager + CursorLoader + LoaderManager.Lo ...

  7. android实现异步加载图片类

    其中牵涉到的关键知识点 1,回调机制,不过回调接口的实现方式有多种多样,可以是一个类继承该接口,也可以是作为一个方法参数: 可以参照自己的这篇博客: http://www.cnblogs.com/bo ...

  8. UE4:四种加载资源的方式

    转自:https://blog.csdn.net/zhangxsv123/article/details/79707686 第一种: 如果该蓝图有C++类(或者说是从C++类创建的蓝图),直接进行加载 ...

  9. Unity3D_异步加载场景(进度条)

    创建两个场景:现在的场景“NowScene”,要加载的场景“LoadScene”: “NowScene”如图所示,“LoadScene”任意: 创建脚本“AsyncLoadScene”,复制如下代码, ...

随机推荐

  1. bzoj 1113 [Poi2008]海报PLA 单调栈

    [Poi2008]海报PLA Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1304  Solved: 896[Submit][Status][Dis ...

  2. tp查找某字段,排除某字段,不用一次写那么多

    更多的情况下我们都是查询某些字段,但有些情况下面我们需要通过字段排除来更方便的查询字段,例如文章详细页,我们可能只需要排除status和update_time字段,这样就不需要写一堆的字段名称了(有些 ...

  3. C# 后台获取请求来源、文件下载

    文件流下载文件 void BigFileDownload() { try { string FileName = "测试.docx"; string filePath = Page ...

  4. Spring Security 过滤器链

    Alias Filter Class Namespace Element or Attribute CHANNEL_FILTER ChannelProcessingFilter http/interc ...

  5. py_faster_rcnn识别出来的结果好多红框重叠

    py_faster_rcnn识别出来的结果好多红框重叠, 可以通过调节demo.py中的NMS_THRESH的值进行限制. NMS_THRESH表示非极大值抑制,这个值越小表示要求的红框重叠度越小,0 ...

  6. Vuejs - 单文件组件

    为什么需要单文件组件 在之前的实例中,我们都是通过 Vue.component 或者 components 属性的方式来定义组件,这种方式在很多中小规模的项目中还好,但在复杂的项目中,下面这些缺点就非 ...

  7. 6、MySQL索引种类

    1.普通索引 这是最基本的索引,它没有任何限制,比如上文中为title字段创建的索引就是一个普通索引,MyIASM中默认的BTREE类型的索引,也是我们大多数情况下用到的索引. –直接创建索引 CRE ...

  8. hdu 1599 find the mincost route (最小环与floyd算法)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1599 find the mincost route Time Limit: 1000/2000 MS ...

  9. python进行机器学习(三)之模型选择与构建

    Scikit-Learn库已经实现了所有基本机器学习的算法,可以直接调用里面库进行模型构建. 一.逻辑回归 大多数情况下被用来解决分类问题(二元分类),但多类的分类(所谓的一对多方法)也适用.这个算法 ...

  10. 【shell】shell中各种括号的作用()、(())、[]、[[]]、{}

    一.小括号,圆括号() 1.单小括号 ()    ①命令组.括号中的命令将会新开一个子shell顺序执行,所以括号中的变量不能够被脚本余下的部分使用.括号中多个命令之间用分号隔开,最后一个命令可以没有 ...