HelloWorldScene解析

2024-01-22 21:08
文章标签 解析 helloworldscene

本文主要是介绍HelloWorldScene解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

http://blog.sina.com.cn/s/blog_9c3d739101017d2n.html

1. 导演类

VS新建一个CS-X工程,然后自己帮我们生成一个HelloWorld工程,然后看看AppGelegate.app文件,里面有这个函数:

bool AppDelegate::applicationDidFinishLaunching()

{

    // initialize director

    CCDirector *pDirector = CCDirector::sharedDirector();

    pDirector->setOpenGLView(CCEGLView::sharedOpenGLView());

 

    // enable High Resource Mode(2x, such as iphone4) and maintains low resource on other devices.

//     pDirector->enableRetinaDisplay(true);

 

    // turn on display FPS

    pDirector->setDisplayStats(true);

 

    // set FPS. the default value is 1.0/60 if you don't call this

    pDirector->setAnimationInterval(1.0 / 60);

 

    // create a scene. it's an autorelease object

    CCScene *pScene = HelloWorld::scene();

 

    // run

    pDirector->runWithScene(pScene);

    return true;

}

因为我还是新手,只能从表面上来理解。

 

CCDirector类,是导演类,在CS-X的里,构建一个游戏,就是构建一个世界(其实我觉得所有的程序都是在构建世界~噗),这个世界由导演类去指导演绎,于是,在打开游戏的第一步时,导演告诉程序:

pDirector->setDisplayStats(true);      // 要显示FPS数值,方便我查看游戏效率

pDirector->setAnimationInterval(1.0 / 60);    // 游戏每秒播放多少帧(这里是60帧)

pDirector->runWithScene(pScene);       // 进入游戏,首先要显示哪个场景

 

于是代码都很简单,主要是要留意,一个类继承了CCScene就可以作为场景使用,然后用导演类的runWithScene方法运行游戏的第一个场景,但是以后切换场景就要用replaceScene方法了。

 

2. 自定义场景

于是现在来看看自定义的场景类,先看看HelloWorldScene.h头文件,很短,看着舒服~

class HelloWorld : public cocos2d::CCLayer

{

public:

    virtual bool init(); 

    static cocos2d::CCScenescene();

 

    void menuCloseCallback(CCObjectpSender);

 

    CREATE_FUNC(HelloWorld);

};

menuCloseCallback函数我觉得可以暂时不理它,算了,还是忍不住要理一下。它是一个回调函数,没了,仅此而已,就是AndroidOnClickListener里面的onClick函数一样,但是又有点不一样,糟糕,我不解释了,乱了,这不是重点~

 

scene函数是不可缺的,它是CCLayer的类,HelloWorld里要重写这个函数的。那scene函数在哪个地方会用到?还记得之前那段代码吗?新建一个场景类的时候是这样的:

// create a scene. it's an autorelease object

    CCScene *pScene = HelloWorld::scene();

 

    // run

    pDirector->runWithScene(pScene);

好了,就是这样的。

 

来看看CREATE_FUNC(HelloWorld);这个我纠结了很久,CREATE_FUNC是一个宏(应该是的,我对C++只停留在初级语法):

#define CREATE_FUNC(__TYPE__) \

static __TYPE__create() \

{ \

    __TYPE__ *pRet = new __TYPE__(); \

    if (pRet && pRet->init()) \

    { \

        pRet->autorelease(); \

        return pRet; \

    } \

    else \

    { \

        delete pRet; \

        pRet = NULL; \

        return NULL; \

    } \

}

CREATE_FUNC实际上就是create函数,准确的说,是重写了CCLayer里的create函数,看看CCLayercreate函数:

CCLayer *CCLayer::create()

{

    CCLayer *pRet = new CCLayer();

    if (pRet && pRet->init())

    {

        pRet->autorelease();

        return pRet;

    }

    else

    {

        CC_SAFE_DELETE(pRet);

        return NULL;

    }

}

是的,CREATE_FUNC的作用只不过是把create的返回值改成继承了CCLayer的实际子类(有点晕,仔细逻辑一下~

 

好了,我假设我们已经懂了CREATE_FUNC的作用了,不~!还没呢~!看看

HelloWorldScene.cpp里是怎么重写scene函数的:

CCSceneHelloWorld::scene()

{

    CCScene * scene = NULL;

    do

    {

        // 'scene' is an autorelease object

        scene = CCScene::create();

        CC_BREAK_IF(! scene);

 

        // 'layer' is an autorelease object

        HelloWorld *layer = HelloWorld::create();

        CC_BREAK_IF(! layer);

 

        // add layer as a child to scene

        scene->addChild(layer);

    } while (0);

 

    // return the scene

    return scene;

}

 

于是我们发现了,虽然HelloWorld自称是场景类,但是导演要的并不是它~!导演还是要了一个真正的CCScene类,不信你看:

            scene = CCScene::create();

这里创建了一个真正的CCScene类。

 

还不够,再看:

    HelloWorld *layer = HelloWorld::create();

    scene->addChild(layer);

HelloWorld只是作为一个layer(一个层),它被添加到刚刚创建的那个CCScene类了。好吧,现实很残酷,尽管HelloWorld基础了场景类,可它依旧不是一个真正的场景~!

 

,紧接着,不要放松,还记得刚刚create函数里有这几句吗?

if (pRet && pRet->init())

    {

        pRet->autorelease();

        return pRet;

    }

 

OK,我知道大家还记得的,于是刚刚拿两句又还记得吗?

HelloWorld *layer = HelloWorld::create();

scene->addChild(layer);

 

是的,也就是说,场景类的init函数是一定会被调用的。

于是,这就表示,继承了CCScene的类,必须要有这三个家伙:

public:

    virtual bool init(); 

    static cocos2d::CCScenescene();

    CREATE_FUNC(HelloWorld);

};

 

3. 最后的最后,当然是总结

很混乱,我自己也很混乱,原来我的表达能力(噗,我自己也无法原谅自己啊= =),所以要总结一下。

1). HelloWorld工程主要就是教我们创建一个场景类

2). AppDelegate.cpp里会创建第一个显示的场景类,此时用的方法是runWithScene

3).要新建一个场景类,就一定要有以下三个函数:

virtual bool init();                // 初始化,可以在这里创建精灵

   

static cocos2d::CCScenescene();  // 在这里要自己新建一个CCScene类,然后把我们自定义的场景类作为一个CCLayer添加到这个CCScene类中

   

CREATE_FUNC(HelloWorld);           // 相当于重写create函数,让create函数返回子类对象。

4. 然后,就是这样了。

 

好了,以上就是我从HelloWorld工程能够发现的东西了,肯定会有不少错误的思想,但是现在的我还无法修正。所以,仅供参考,嘿嘿~

这篇关于HelloWorldScene解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/634237

相关文章

MySQL字符串转数值的方法全解析

《MySQL字符串转数值的方法全解析》在MySQL开发中,字符串与数值的转换是高频操作,本文从隐式转换原理、显式转换方法、典型场景案例、风险防控四个维度系统梳理,助您精准掌握这一核心技能,需要的朋友可... 目录一、隐式转换:自动但需警惕的&ld编程quo;双刃剑”二、显式转换:三大核心方法详解三、典型场景

SQL 注入攻击(SQL Injection)原理、利用方式与防御策略深度解析

《SQL注入攻击(SQLInjection)原理、利用方式与防御策略深度解析》本文将从SQL注入的基本原理、攻击方式、常见利用手法,到企业级防御方案进行全面讲解,以帮助开发者和安全人员更系统地理解... 目录一、前言二、SQL 注入攻击的基本概念三、SQL 注入常见类型分析1. 基于错误回显的注入(Erro

C++ 多态性实战之何时使用 virtual 和 override的问题解析

《C++多态性实战之何时使用virtual和override的问题解析》在面向对象编程中,多态是一个核心概念,很多开发者在遇到override编译错误时,不清楚是否需要将基类函数声明为virt... 目录C++ 多态性实战:何时使用 virtual 和 override?引言问题场景判断是否需要多态的三个关

Springboot主配置文件解析

《Springboot主配置文件解析》SpringBoot主配置文件application.yml支持多种核心值类型,包括字符串、数字、布尔值等,文章详细介绍了Profile环境配置和加载位置,本文... 目录Profile环境配置配置文件加载位置Springboot主配置文件 application.ym

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

一文解析C#中的StringSplitOptions枚举

《一文解析C#中的StringSplitOptions枚举》StringSplitOptions是C#中的一个枚举类型,用于控制string.Split()方法分割字符串时的行为,核心作用是处理分割后... 目录C#的StringSplitOptions枚举1.StringSplitOptions枚举的常用

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

MyBatis延迟加载与多级缓存全解析

《MyBatis延迟加载与多级缓存全解析》文章介绍MyBatis的延迟加载与多级缓存机制,延迟加载按需加载关联数据提升性能,一级缓存会话级默认开启,二级缓存工厂级支持跨会话共享,增删改操作会清空对应缓... 目录MyBATis延迟加载策略一对多示例一对多示例MyBatis框架的缓存一级缓存二级缓存MyBat