Cocos2d-x 启动过程详解:渲染

Qiu Yang2014-08-21 09:36:001417 次阅读

本文主要讲解Cocos2d-x的整体启动过程:Cocos2d-x 在各个平台的实现代码是一样的,只要针对不同平台做相应的配置就可以了。


一、启动前奏

现在来看一下在iOS平台下的相关结构:


打开源代码自带工程,你会看到一个main文件,这里main里面有一个main函数,这是程序的入口函数。在这里它会加载AppController,进入这个类,这里有iOS平台初始化代码,但是最先执行的如下:

1
2
//cocos2dapplicationinstance
static AppDelegates_sharedApplication;


在这里Cocos2d-x创建了一个appDelegare的对象,当然在创建的过程中会进行相应的初始化,通过代码可以看到:

2
3
4
5
/**
@briefThecocos2dApplication.
ThereasontoimplementwithprivateinheritanceistohidesomeinterfacedetailsofCCDirector.
*/
class AppDelegate: private cocos2d::CCApplication


这里就会调用CCApplicaiton的构造函数:

5
CCApplication::CCApplication()
{
CC_ASSERT(!sm_pSharedApplication);
sm_pSharedApplication= this ; //全局共享实例对象
}

将this赋值给了sm_pSharedApplication, 这个this是什么? 实际上this是AppDeletegate, 因为这里的调用过程中并没有涉及到父类的对象,如果要涉及应该是CCApplication::CCApplication(); C++中,在创建派生类对象的时候不会创建父类对象,只会显示或者隐式的调用父类的构造函数。


但是这里为什么会有一个没有用到的全局变量呢? 答案在后面:

1
cocos2d::CCApplication::sharedApplication()->run();

一开始定义一个static 变量就是要定义一个CCApplication的实例,然后调用run函数。


进入run函数:

5
6
7
8
int CCApplication::run()
{
if (applicationDidFinishLaunching())
{
[[CCDirectorCallersharedDirectorCaller]startMainLoop];
}
return 0;
}

ok,可以看到,当applicationDidFinishLaunching执行成功后就会执行startMainLoop函数(这其实就启动了线程),分别看一下源代码:


applicationDidFinishLaunching 是CCApplication的抽象函数,在AppDelegate中实现:

8
9
10
11
12
13
14
15
bool AppDelegate::applicationDidFinishLaunching()
//initializedirector
CCDirector*pDirector=CCDirector::sharedDirector();
pDirector->setopenGLView(CCEGLView::sharedOpenGLView());
//turnondisplayFPS
pDirector->setdisplayStats( true );
//setFPS.thedefaultvalueis1.0/60ifyoudon’tcallthis
pDirector->setAnimationInterval(1.0/60);
//createascene.it’sanautoreleaSEObject
CCScene*pScene=HelloWorld::scene();
//run
pDirector->runWithScene(pScene);
return ;
}

其实对于游戏运行来说,说白了就是一个死循环,就像win的消息那样。


来看一下startMainLoop函数:

9
-( void )startMainLoop
//CCDirector::setAnimationInterval()iscalled,weshouldinvalidateitfirst
[displayLinkinvalidate];
displayLink=nil;
displayLink=[NSClassFromString(@ "CAdisplayLink" )displayLinkWithTarget:selfselector:@selector(doCaller:)];
[displayLinksetFrameInterval:self.interval];
[displayLinkaddToRunLoop:[NSRunLoopcurrentRunLoop]forMode:NSDefaultRunLoopMode];
}

什么?没有循环? 你错了,注意的是这里加载了一个类:CAdisplayLink,就是这里循环起来的,这其实是一个定时器,默认情况是每秒运行60次。CAdisplayLink是一个定时器类,他能以特定的模式注册到runloop,这样每当屏幕显示内容刷新结束,runloop就会向CAdisplayLink指定的target发送一次执行的selector消息,对应的毁掉函数就会调用起来。


可以看到这里有一个回调函数:

4
)doCaller:(id)sender
cocos2d::CCDirector::sharedDirector()->mainLoop();
}


二、开始执行

在这里,我们就进入了mainLoop函数了。这里就是我们的头了:

14
CCdisplayLinkDirector::mainLoop( )
(m_bPurgeDirecotorInNextLoop) //这个变量在end函数里会被设置成为true
m_bPurgeDirecotorInNextLoop= false ;
purgeDirector();
}
else (!m_bInvalid)
{
drawScene();
//releasetheobjects
CCPoolManager::sharedPoolManager()->pop(); //每次回调都会清楚pool池(还记得之前的内存管理么)
}
}

好吧,这里主要的函数drawScene(),来看看主要的实现点:

15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
//DrawtheScene
CCDirector::drawScene( )
{
//calculate“global”dt
calculateDeltaTime(); //距离上次mainloop的时间
//tickbeforeglClear:issue#533
(!m_bPaused) //暂停
{
m_pScheduler->update(m_fDeltaTime); //下面会解释这里的内容
}
glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT); //函数的作用是用当前缓冲区清除值,也就是glClearColor或者glClearDepth、glClearIndex、glClearStencil、glClearaccum等函数所指定的值来清除指定的缓冲区,也可以使用glDrawBuffer一次清除多个颜色缓存。比如:glClearColor(0.0,0.0,0.0,0.0);glClear(GL_COLOR_BUFFER_BIT);第一条语句表示清除颜色设为黑色,第二条语句表示实际完成了把整个窗口清除为黑色的任务,glClear()的唯一参数表示需要被清除的缓冲区
/*toavoidflickr,nextSceneMUSTbehere:aftertickandbeforedraw.
XXX:Whichbugisthisone.Itseemsthatitcan’tbereproducedwithv0.9*/
(m_pNextScene)
{
setNextScene(); //初始化场景,如:onEnter或者onExit
}
kmGLPushmatrix(); //矩阵压栈
//drawthescene
(m_pRunningScene)
{
m_pRunningScene->visit(); //遍历正在运行的Scene,先遍历的是CCNode的visit,因为CCScene并没有实现这个方法。
}
//drawthenotificationsnode
(m_pNotificationNode)
{
m_pNotificationNode->visit(); //遍历m_pNotificationNode,可以通过该m_pNotificationNode绘制特殊的节点
}
(m_bdisplayStats)
{
showStats(); //左下角的fps(每秒传输的帧数)
}
kmGLPopMatrix(); //从矩阵栈中弹出
m_uTotalFrames++;
//swapbuffers
(m_pobOpenGLView)
{
m_pobOpenGLView->swapBuffers();
}
(m_bdisplayStats)
{
calculateMPF();
}
}


我们先一步一步的看,在CCDirector中,mm文件里面有一个句:

CCdisplayLinkDirector*s_SharedDirector=NULL;

在这里CCSisplayLinkDirector是继承自CCDirector的,所以我们在运行的时候会调用:

9
CCDirector*CCDirector::sharedDirector( (!s_SharedDirector)
s_SharedDirector= new CCdisplayLinkDirector();
s_SharedDirector->init();
s_SharedDirector;
}

而后:

11
CCDirector::init( cclOG(“cocos2d:%s”,cocos2dVersion());
…………………
//scheduler初始化CCScheduler定时调度器第二篇中会看到哟
m_pScheduler= CCScheduler();
………………….
//createautoreleasepool这里CCPoolManager管理多个CCAutoreleasePool,将CCAutoreleasePool放到CCPoolManager中的m_pReleasePoolStack
CCPoolManager::sharedPoolManager()->push();
;
三、渲染开始

还记得刚刚我们看到的visit么? 他是在CCNode中实现的,我们来看一下:

45
46
47
48
49
50
51
52
53
54
55
CCNode::visit()
//quickreturnifnotvisible.childrenwon’tbedrawn.
(!m_bVisible) //如果不可见,就会直接返回,不递归迭代
{
;
//压入矩阵
(m_pGrid&&m_pGrid->isActive()) //处理CCGridBase
{
m_pGrid->beforeDraw();
}
->transform(); //这里的工作我们稍后还要继续看
CCNode*pNode=NULL;
unsigned i=0;
(m_pChildren&&m_pChildren->count()>0)
{
sortAllChildren();
//drawchildrenzOrder<0
ccArray*arrayData=m_pChildren->data;
for (;i<arrayData->num;i++) //遍历zOrder<0的子类
pNode=(CCNode*)arrayData->arr[i];
(pNode&&pNode->m_nZOrder<0)
{
pNode->visit();
}
else
{
break ;
}
}
//selfdraw
->draw();
(;i<arrayData->num;i++)
{
pNode=(CCNode*)arrayData->arr[i];
(pNode)
{
pNode->visit();
}
}
}
else
{
->draw();
}
//resetfornextframe
m_uOrderOfArrival=0;
(m_pGrid&&m_pGrid->isActive())
{
m_pGrid->afterDraw( );
}
//出矩阵
}


刚刚看到了transform,这是做什么的呢?它是进行坐标系的变换,没有坐标系的变换,则无法在正确的位置绘制出纹理。变换矩阵等价于坐标系变换.变换矩阵是如何根据当前节点的位置、旋转角度和缩放比例等属性计算出来的了。形象地讲,transform 方法的任务就是根据当前节点的属性计算出如何把绘图坐标系变换为新坐标系的矩阵。


那就继续看transform吧:

20
CCNode::transform()
kmMat4transfrom4x4;
//Convert3×3into4×4matrix
CCAffineTransformtmpAffine= ->nodetoParentTransform(); //获取节点相对于父节点的变换矩阵
CGAffinetoGL(&tmpAffine,transfrom4x4.mat); //转换
//UpdateZvertexmanually
transfrom4x4.mat[14]=m_fVertexZ; //设置z
kmGLMultMatrix(&transfrom4x4); //右乘一个矩阵参数里面那个
//XXX:Expensivecalls.Camerashouldbeintegratedintothecachedaffinematrix
(m_pCamera!=NULL&&!(m_pGrid!=NULL&&m_pGrid->isActive()))
{
translate=(m_obAnchorPointInPoints.x!=0.0f||m_obAnchorPointInPoints.y!=0.0f);
(translate)
kmGLTranslatef(RENDER_IN_SUBPIXEL(m_obAnchorPointInPoints.x),RENDER_IN_SUBPIXEL(m_obAnchorPointInPoints.y),0);
m_pCamera->locate();
(translate)
kmGLTranslatef(RENDER_IN_SUBPIXEL(-m_obAnchorPointInPoints.x),RENDER_IN_SUBPIXEL(-m_obAnchorPointInPoints.y),0);
}
}


一般来说游戏中会大量使用旋转,缩放,平移等仿射变换( 所谓仿射变换是指在线性变换的基础上加上平移,平移不是线性变换)。2D计算机图形学中的仿射变换通常是通过和3×3齐次矩阵相乘来实现的。Cocos2d中的仿射变换使用了Quartz 2D中的CGAffineTransform类来表示:

struct CCAffineTransform{
float a,b,c,d;
tx,ty;
}; typedef CGAffineTransformCGAffineTransform;

在Cocos2d-x中,绘制是使用了openies的,所以CGAffineTransform只是用来表示2d仿射变换的,最终还是要转化成OpenglES的4*4变换矩阵的,因为OpenGL是3d的。这个转换工作是由 CGAffinetoGL来完成的。

变换后的矩阵关系如下:

1
|m11m21m31m41||ac0tx||m12m22m32m42||bd0ty||m13m23m33m43|<=>|0010||m14m24m34m44||0001|


最后来看一个概念:

“节点坐标系”指的是以一个节点作为参考而产生的坐标系,换句话说,它的任何一个子节点的坐标值都是由这个坐标系确定的,通过以上方法,我们可以方便地处理触摸点,也可以方便地计算两个不同坐标系下点之间的方向关系。


这里是整个启动过程的一部分,后面我们还会根据Cocos2d-x的内存机制和回调机制来进行分析,也会有一些深层次的渲染知识。


来源网址:http://iqll.sinaapp.com/cocos2dx-启动过程详解一:渲染/

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

相关推荐


如何使用CCRenderTexture创建动态纹理 Cocos2d-x 2 1 4
    本文实践自 RayWenderlich、Ali Hafizji 的文章《How To Create Dynamic Textures with CCRenderTexture in Cocos2D 2.X》,文中使用Cocos2D,我在这里使用Cocos2D-x 2.1.4进行学习和移植。在这篇文章,将会学习到如何创建实时纹理、如何用Gimp创建无缝拼接纹
Cocos-code-ide使用入门学习
Cocos-code-ide使用入门学习地点:杭州滨江邮箱:appdevzw@163.com微信公众号:HopToad 欢迎转载,转载标注出处:http://blog.csdn.netotbaron/article/details/424343991.  软件准备 下载地址:http://cn.cocos2d-x.org/download 2.  简介2.1         引用C
Cocos2D-x-3.0 编译(Win7)
第一次開始用手游引擎挺激动!!!进入正题。下载资源1:从Cocos2D-x官网上下载,进入网页http://www.cocos2d-x.org/download,点击Cocos2d-x以下的Download  v3.0,保存到自定义的文件夹2:从python官网上下载。进入网页https://www.python.org/downloads/,我当前下载的是3.4.0(当前最新
quick-cocos2d-x实例之挑战记忆极限设计文档
1.  来源 QuickV3sample项目中的2048样例游戏,以及最近《最强大脑》娱乐节目。将2048改造成一款挑战玩家对数字记忆的小游戏。邮箱:appdevzw@163.com微信公众号:HopToadAPK下载地址:http://download.csdn.net/detailotbaron/8446223源码下载地址:http://download.csdn.net/
Cocos2d-x 3 X CMake MinGW版本编译运行
   Cocos2d-x3.x已经支持使用CMake来进行构建了,这里尝试以QtCreatorIDE来进行CMake构建。Cocos2d-x3.X地址:https://github.com/cocos2d/cocos2d-x1.打开QtCreator,菜单栏→"打开文件或项目...",打开cocos2d-x目录下的CMakeLists.txt文件;2.弹出CMake向导,如下图所示:设置
vs 2013 编译cocos2d-x-3.9
 下载地址:链接:https://pan.baidu.com/s/1IkQsMU6NoERAAQLcCUMcXQ提取码:p1pb下载完成后,解压进入build目录使用vs2013打开工程设置平台工具集,打开设置界面设置: 点击开始编译等待编译结束编译成功在build文件下会出现一个新文件夹Debug.win32,里面就是编译
Cocos2d-x CCControlPotentiometer之圆形音量button及特效
1. 圆形音量button事实上作者的本意应该是叫做“电位计button”。可是我觉得它和我们的圆形音量button非常像,所以就这么叫它吧~先看效果:好了,不多解释,本篇到此为止。(旁白: 噗。就这样结束了?)啊才怪~我们来看看代码:[cpp] viewplaincopyprint?CCContro
Cocos2d-x入门教程二简单的静态显示对象
原文链接:http://www.cnblogs.com/physwf/archive/2013/04/26/3043912.html为了进一步深入学习贯彻Cocos2d,我们将自己写一个场景类,但我们不会走的太远,凡是都要循序渐进,哪怕只前进一点点,那也至少是前进了,总比贪多嚼不烂一头雾水的好。在上一节中我们建
  • • 如何使用CCRenderTexture创建动态纹理 …
  • • Cocos-code-ide使用入门学习
  • • Cocos2D-x-3.0 编译(Win7)
  • • Cocos2d-x 2 0 在Windows平台下的使用
  • • quick-cocos2d-x实例之挑战记忆极限设计…
  • • Cocos2d-x 3 X CMake MinGW版本编译运行
  • • vs 2013 编译cocos2d-x-3.9
  • • cocos2d-x游戏开发系列教程-超级玛丽01…
  • • Cocos2d-x CCControlPotentiometer之圆…
  • • Cocos2d-x入门教程二简单的静态显示对象
  • • cocos2d-x中CCScale9Sprite的另一种实现
  • • Cocos2d-x v2.2.2版本+Win7+VS2010环境…
  • • Ubuntu14.04+eclipse下cocos2d-x3.0正式…
  • • 分别基于WIN32 API界面编程和Cocos2d-x…
  • • Cocos2d-x 开发小记二:控件
  • • Cocos2d-x 开发小记一:基本动作
  • • 买Cocos2d-x视频课程送纸质图书
  • • ‎Cocos2d-x 学习笔记(11.10) Spawn

Cocos2d-x 启动过程详解:渲染的更多相关文章

  1. HTML利用九宫格原理进行网页布局

    这篇文章主要介绍了HTML利用九宫格原理进行网页布局,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  2. ios – 围绕x轴旋转AVAssetWriter的输出180度

    我正在使用AVAssetWriter创建一个Quicktime电影文件.目前输出视频是“倒置”.理论上,我可以通过围绕水平轴旋转180度来纠正这个问题.最好的方法是什么?Appledocs和wikipedia都没有明确说明仿射变换矩阵是如何工作的.并且可能有更好的方式.解决方法如果要围绕z轴旋转视频180度,或者如果你想在x轴上反射

  3. .dylib在Debug中链接,在XCode中找不到适用于iPhone的版本

    所以我已经将libxml2.2.dylib库包含在我的iPhoneXCode项目中,以创建一些Xml和XPath解析实用程序.当我编译并运行在模拟器和设备的调试模式时,我没有问题,但是,当我切换到发布模式我得到…

  4. 在编译的iOS应用程序(IPA)中加密内容

    由于IPA结构只是一个压缩文件,包含编译代码媒体内容,如图像&音频,我如何保护内容免受别人的窃取?是否有加密可以添加到IPA?

  5. ios – Swift 4向后兼容性

    一起使用.有没有办法在两个版本的Xcode中使这个工作?Swift4是否应该向后兼容?

  6. Swift与OC混合编译

    SWift调用OC新建swift文件此时系统自动生成-Bridging-Header.h文件并且TARGETS->BuildSettings->Objective-CBridgingHeader(搜索bridg)选项中会自动填入以上头文件的路径在-Bridging-Header.h中#import要调用的OC对象头文件OC调用Swift在OC文件中#import“

  7. Swift教程01-使用switfc终端命令编译运行swift程序

    应公司的要求,要我研究swift语言,然后给大家进行swift技术培训,买了4,5本swift相关的书籍就开始研究了.今天来介绍一下,swift相关的终端的命令.1.首先我们在桌面上建一个swift文件夹用来存放.swift源文件,打开终端输入cd加空格拖拽文件夹到终端(使用命令ls,cd目录也是等效)3.使用vi命令创建一个HelloWorld.swift源文件进入swift文件夹后,然后使用如

  8. 驳狗屎文 "我为什么放弃Go语言"

    开门见山地说,我当初放弃Go语言,就是因为两个“不爽”:第一,对Go语言本身不爽;第二,对Go语言社区里的某些人不爽。两年之后,2011年底,Go语言发布1.0的计划被提上日程,相关的报道又多起来,我再次关注它,重新评估之后决定深入参与Go语言。外加Go语言社区里的某些人,其中也包括Google公司负责开发Go语言的某些人,其态度、言行,让我极度厌恶,促使我决绝地离弃Go语言。第1节:我为什么对Go语言不爽?

  9. Swift语法特点

    从它的语法中能看到Ojective-CJavaScriptC#Python等语言的影子swift借鉴了以上几门语言的特点我们可以从swift身上看到这几种语言的影子同时还具备了编译型语言额高能性和脚本语言的灵活交互性OC-C#都是编译型的语言也就是我们写完代码要保证每一行代码编译通过才能跑起来这叫编译型语言一般编译型的语言性能比较高能做的事情比较强大比如说OC能开发手机程序C#能开发服务器程序脚本

  10. Swift 2.0学习笔记Day 35——会使用下标吗?

    下标Swift中的下标相当于Java中的索引属性和C#中的索引器。getter访问器是一个方法,在最后使用return语句将计算结果返回。setter访问器“新属性值”是要赋值给属性值。参数的声明可以省略,系统会分配一个默认的参数newValue。可以自定义一个二维数组类型,然后通过两个下标参数访问它的元素,形式上类似于C语言的二维数组。

随机推荐

  1. 【cocos2d-x 3.x 学习笔记】对象内存管理

    Cocos2d-x的内存管理cocos2d-x中使用的是上面的引用计数来管理内存,但是又增加了一些自己的特色。cocos2d-x中通过Ref类来实现引用计数,所有需要实现内存自动回收的类都应该继承自Ref类。下面是Ref类的定义:在cocos2d-x中创建对象通常有两种方式:这两中方式的差异可以参见我另一篇博文“对象创建方式讨论”。在cocos2d-x中提倡使用第二种方式,为了避免误用第一种方式,一般将构造函数设为protected或private。参考资料:[1]cocos2d-x高级开发教程2.3节[

  2. 利用cocos2dx 3.2开发消灭星星六如何在cocos2dx中显示中文

    由于编码的不同,在cocos2dx中的Label控件中如果放入中文字,往往会出现乱码。为了方便使用,我把这个从文档中获取中文字的方法放在一个头文件里面Chinese.h这里的tex_vec是cocos2dx提供的一个保存文档内容的一个容器。这里给出ChineseWords,xml的格式再看看ChineseWord的实现Chinese.cpp就这样,以后在需要用到中文字的地方,就先include这个头文件然后调用ChineseWord函数,获取一串中文字符串。

  3. 利用cocos2dx 3.2开发消灭星星七关于星星的算法

    在前面,我们已经在GameLayer中利用随机数初始化了一个StarMatrix,如果还不知道怎么创建星星矩阵请回去看看而且我们也讲了整个游戏的触摸事件的派发了。

  4. cocos2dx3.x 新手打包APK注意事项!

    这个在编译的时候就可以发现了比较好弄这只是我遇到的,其他的以后遇到再补充吧。。。以前被这两个问题坑了好久

  5. 利用cocos2dx 3.2开发消灭星星八游戏的结束判断与数据控制

    如果你看完之前的,那么你基本已经拥有一个消灭星星游戏的雏形。开始把剩下的两两互不相连的星星消去。那么如何判断是GameOver还是进入下一关呢。。其实游戏数据贯穿整个游戏,包括星星消除的时候要加到获得分数上,消去剩下两两不相连的星星的时候的加分政策等,因此如果前面没有做这一块的,最好回去搞一搞。

  6. 利用cocos2dx 3.2开发消灭星星九为游戏添加一些特效

    needClear是一个flag,当游戏判断不能再继续后,这个flag变为true,开始消除剩下的星星clearSumTime是一个累加器ONE_CLEAR_TIME就是每颗星星消除的时间2.连击加分信息一般消除一次星星都会有连击信息和加多少分的信息。其实这些combo标签就是一张图片,也是通过控制其属性或者runAction来实现。源码ComboEffect.hComboEffect.cpp4.消除星星粒子效果消除星星时,为了实现星星爆裂散落的效果,使用了cocos2d提供的粒子特效引擎对于粒子特效不了

  7. 02 Cocos2D-x引擎win7环境搭建及创建项目

    官网有搭建的文章,直接转载记录。环境搭建:本文介绍如何搭建Cocos2d-x3.2版本的开发环境。项目创建:一、通过命令创建项目前面搭建好环境后,怎样创建自己的Cocos2d-x项目呢?先来看看Cocos2d-x3.2的目录吧这就是Cocos2d-x3.2的目录。输入cocosnew项目名–p包名–lcpp–d路径回车就创建成功了例如:成功后,找到这个项目打开proj.win32目录下的Hello.slnF5成功了。

  8. 利用cocos2dx 3.2开发消灭星星十为游戏添加音效项目源码分享

    一个游戏,声音也是非常的重要,其实cocos2dx里面的简单音效引擎的使用是非常简单的。我这里只不过是用一个类对所有的音效进行管理罢了。Audio.hAudio.cpp好了,本系列教程到此结束,第一次写教程如有不对请见谅或指教,谢谢大家。最后附上整个项目的源代码点击打开链接

  9. 03 Helloworld

    程序都有一个入口点,在C++就是main函数了,打开main.cpp,代码如下:123456789101112131415161718#include"main.h"#include"AppDelegate.h"#include"cocos2d.h"USING_NS_CC;intAPIENTRY_tWinMain{UNREFERENCED_ParaMETER;UNREFERENCED_ParaMETER;//createtheapplicationinstanceAppDelegateapp;return

  10. MenuItemImage*图标菜单创建注意事项

    学习cocos2dx,看的是cocos2d-x3.x手游开发实例详解,这本书错误一大把,本着探索求知勇于发现错误改正错误的精神,我跟着书上的例子一起调试,当学习到场景切换这个小节的时候,出了个错误,卡了我好几个小时。

返回
顶部