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. lightoj 1282 && uva 11029

    Leading and Trailing lightoj 链接:http://lightoj.com/volume_showproblem.php?problem=1282 uva 链接:http:/ ...

  2. HDU2732 最大流

    Leapin' Lizards Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Tot ...

  3. FreeRTOS - 定时器使用注意

    1.只有进入定时器守护任务,从定时器命令队列取出命令,队列空间才会空出一个可用空间:所有定时器公用一个定时器队列 2.如果使用软件定时器,在调度器开始前,会自动创建一个定时器守护任务,configTI ...

  4. Jenkins自动化构建系列:01敏捷开发、自动化构建与持续集成

    <SVN与TortoiseSVN实战系列>已写完,今天新开一个<Jenkins自动化构建系列>,上周听了Bob Jiang老师的Agile1001公开课,一直想写个总结,这篇关 ...

  5. 查看自己电脑外网IP

    连着wifi,在CMD窗口中显示的无线局域网适配器IP很有可能是内网IP.此时可以通过下面方法查看自己的电脑外网IP. 1.前提条件可以上外网: 2.上外网百度,输入IP,进行搜索. 3.查看结果即可 ...

  6. 【设计模式】 模式PK:命令模式VS策略模式

    1.概述 命令模式和策略模式的类图确实很相似,只是命令模式多了一个接收者(Receiver)角色.它们虽然同为行为类模式,但是两者的区别还是很明显的.策略模式的意图是封装算法,它认为“算法”已经是一个 ...

  7. 【C++对象模型】第四章 Function 语意学

    1.Member的各种调用方式 1.1 Nonstatic Member Functions 实际上编译器是将member function被内化为nonmember的形式,经过下面转化步骤: 1.给 ...

  8. ECMAScript5中新增的Array方法实例详解

    ECMAScript5标准发布于2009年12月3日,它带来了一些新的,改善现有的Array数组操作的方法.(注意兼容性) 在ES5中,一共有9个Array方法:http://kangax.githu ...

  9. 解决vue代码缩进报错问题 关闭ESlint

    前言 使用vue-cli来构建单页SPA应用,提示代码缩进报错 原因分析 通过查看package.json文件我们可以发现,在文件中默认安装了eslint-loader模块,eslint-loader ...

  10. NodeJS中Buffer模块详解

    一,开篇分析 所谓缓冲区Buffer,就是 "临时存贮区" 的意思,是暂时存放输入输出数据的一段内存. JS语言自身只有字符串数据类型,没有二进制数据类型,因此NodeJS提供了一 ...