我的Cocos2d-x学习笔记(九)游戏帧循环(游戏主循环)

游戏运行时候会不断按照游戏逻辑规则重新绘图,反复处理用户输入、处理定时事件、绘图,直到游戏结束。

Cocos2d-x引擎也是通过不断绘图来进行游戏的,默认的帧率在AppDelegate.cpp中显示为60帧每秒,也就是每秒执行了60次用户输入、定时时间、绘图等。

抄录Cocos2d-x高级教程中内容如下:

游戏乃至图形界面的本质是不断地绘图,然而绘图并不是随意的,任何游戏都需要遵循一定的规则来呈现出来,这些规则就体现为游戏逻辑。

游戏逻辑会控制游戏内容,使其根据用户输入和时间流逝而改变。因此,游戏可以抽象为不断地重复以下动作:

处理用户输入

处理定时事件

绘图
      游戏主循环就是这样的一个循环,它会反复执行以上动作,保持游戏进行下去,直到玩家退出游戏。

CCDirector::mainLoop()方法,这个方法负责调用定时器,绘图,发送全局通知,并处理内存回收池。

该方法按帧调用,每帧调用一次,而帧间间隔取决于两个因素,一个是预设的帧率,默认为60 帧每秒;

另一个是每帧的计算量大小。当逻辑处理与绘图计算量过大时,设备无法完成每秒60 次绘制,此时帧率就会降低。

通过Cocos2d-x高级教程中介绍可知,CCDirector::mainLoop()负责调用定时器,绘图,发送全局通知,那么CCDirector::mainLoop()是在那里开始执行的呢?就是在

CCApplication中,CCApplication中创建游戏对象并初始化,之后开始执行游戏主循环,代码如下:

CCApplication * CCApplication::sm_pSharedApplication = 0;
CCApplication* CCApplication::sharedApplication()
{
	return sm_pSharedApplication;
}
int CCApplication::run()
{
	while (1)
	{
		....
		if (!PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
		{
			....
				// If it‘s the time to draw next frame, draw it, else sleep a while.
			if (nNow.QuadPart - nLast.QuadPart > m_nAnimationInterval.QuadPart)
			{
				nLast.QuadPart = nNow.QuadPart;
				CCDirector::sharedDirector()->mainLoop();
			}
			....
		}
		.....
		if (!m_hAccelTable || !TranslateAccelerator(msg.hwnd, m_hAccelTable, &msg))
		{
			TranslateMessage(&msg);
			DispatchMessage(&msg);
		}
	}
	return xxx;
}

上面这段代码就是开始游戏主循环,while中调用了CCDirector::mainLoop(),之后游戏帧循环开始;

另外复习一下,在main.cpp中有调用CCApplication::run():

int APIENTRY _tWinMain(...)
{
	...
	AppDelegate app;
	CCEGLView* eglView = CCEGLView::sharedOpenGLView();
	eglView->setViewName("HelloWorld");
	eglView->setFrameSize(480, 320);
	return CCApplication::sharedApplication()->run();
}

CCApplication::sharedApplication()->run();后游戏开始运行。

之后看看CCDirector::mainLoop()中内容,代码如下:

class CC_DLL CCDirector : public CCObject, public TypeInfo
{
public:
	virtual void mainLoop(void) = 0;
}

在CCDirector中mainLoop为一个纯虚函数,没有具体实现。而具体实现mainLoop的则是在CCDirector子类CCDisplayLinkDirector中,精简后的CCDisplayLinkDirector类如下:

class CCDisplayLinkDirector : public CCDirector
{
public:
	virtual void mainLoop(void);
}
void CCDisplayLinkDirector::mainLoop(void)
{
	if (m_bPurgeDirecotorInNextLoop)
	{
		m_bPurgeDirecotorInNextLoop = false;
		purgeDirector();
	}
	else if (!m_bInvalid)
	{
		drawScene();

		// release the objects
		CCPoolManager::sharedPoolManager()->pop();完成内存清理
	}
}

在CCDisplayLinkDirector中的mainLoop中可以看到有个drawScene()函数,这个函数在程序中完成定时与图片渲染工作,看看drawScene()的定义,如下:

void CCDirector::drawScene(void)
{
	if (!m_bPaused)
	{
		m_pScheduler->update(m_fDeltaTime);
	}
	// draw the scene
	if (m_pRunningScene)
	{
		m_pRunningScene->visit();
	}
}

首先m_pScheduler->update(m_fDeltaTime);这句代码完成了定时器事件的处理;

之后m_pRunningScene->visit();这句代码遍历每个节点,完成渲染工作。

以上就是drawScene()中我们关注的功能;

然后在看看drawScene()下面的CCPoolManager::sharedPoolManager()->pop(),这句代码完成内存清理的功能。

之后再返回CCApplication中继续往下看,有两句代码如下:

int CCApplication::run()
{
	while (1)
	{
		.....
		if (!m_hAccelTable || !TranslateAccelerator(msg.hwnd, m_hAccelTable, &msg))
		{
			TranslateMessage(&msg);
			DispatchMessage(&msg);
		}
	}
	return xxx;
}

TranslateMessage(&msg)和DispatchMessage(&msg)在Win32中完成事件的收集与分发,在这个游戏中则是处理触摸消息。

到此为止,游戏中主循环执行玩一遍,下面精简一下描述,方便以后复习:

CCApplication::applicationDidFinishLaunching();//游戏内容设计
int CCApplication::run()
{
	while (1)
	{	....
		if (!PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
		{	....
				// If it‘s the time to draw next frame, draw it, else sleep a while.
			if (nNow.QuadPart - nLast.QuadPart > m_nAnimationInterval.QuadPart)
			{
				CCDirector::sharedDirector()->mainLoop()
					{
						void CCDirector::drawScene(void)
							{
								m_pScheduler->update(m_fDeltaTime);//定时器事件处理
								m_pRunningScene->visit();//完成图片的渲染
							}
						CCPoolManager::sharedPoolManager()->pop();// 完成内存清理
					}
			}....
		}.....
		if (!m_hAccelTable || !TranslateAccelerator(msg.hwnd, m_hAccelTable, &msg))
		{
			TranslateMessage(&msg);//
			DispatchMessage(&msg);//处理触摸消息
		}
	}
	return xxx;
}
时间: 2024-10-17 02:28:02

我的Cocos2d-x学习笔记(九)游戏帧循环(游戏主循环)的相关文章

APUE 学习笔记(九) 高级I/O

1. 非阻塞I/O 低速系统调用时可能会使进程永远阻塞的一类系统调用,包括以下调用: (1)某些文件类型你(网络socket套接字.终端设备.管道)暂无可使用数据,则读操作可能会使调用者永远阻塞 (2)如果数据不能立即被(1)中文件类型接受,则写操作会使调用者永远阻塞 (3)某些进程间通信函数 非阻塞I/O使我们可以调用open.read.write这样的I/O操作,并使这些操作不会永远阻塞,如果这种操作不能完成,则调用立即出错返回 对于一个给定的文件有两种方法对其指定非阻塞I/O: (1)调用

【Unity 3D】学习笔记三十四:游戏元素——常用编辑器组件

常用编辑器组件 unity的特色之一就是编辑器可视化,很多常用的功能都可以在编辑器中完成.常用的编辑器可分为两种:原有组件和拓展组件.原有组件是编辑器原生的一些功能,拓展组件是编辑器智商通过脚本拓展的新功能. 摄像机 摄像机是unity最为核心组件之一,游戏界面中显示的一切内容都得需要摄像机来照射才能显示.摄像机组件的参数如下: clear flags:背景显示内容,默认的是skybox.前提是必须在render settings 中设置天空盒子材质. background:背景显示颜色,如果没

python学习笔记九——文件与目录

1.python进行文件读写的函数是open或file类 mode:r  只读 r+   读写 w  写入,先删除原文件,再重新写入,如果文件没有则创建 w+  读写,先删除原文件,再重新写入,如果文件没有则创建(可写入和输出) a  写入,在文件末尾追加新的内容,文件不存在则创建 a+  读写,在文件末尾追加新的内容,文件不存在则创建 b  打开二进制文件,可与r,w,a,+结合使用 U  支持所有的换行符号,"\r","\n","\r\n"

angular学习笔记(九)-css类和样式3

再来看一个选择li列表的例子: 点击li中的任意项,被点击的li高亮显示: <!DOCTYPE html> <html ng-app> <head> <title>6.3css类和样式</title> <meta charset="utf-8"> <script src="../angular.js"></script> <script src="scri

angular学习笔记(九)-css类和样式2

在上一个例子中,元素的类名使用拼接的方法,这样,类名中就不得不带有true或false,并且不易维护,所以,angular使用ng-class属性来控制元素的类名: 我们来看一个小例子,点击error按钮,顶部提示错误框,点击warning按钮,顶部提示警告框. 错误框的类名是.err,警告框的类名是.warn: <!DOCTYPE html> <html ng-app> <head> <title>6.2css类和样式</title> <

【Unity 3D】学习笔记三十二:游戏元素——游戏光源

游戏光源 在3D游戏中,光源是一个非常具有特色的游戏组件.用来提升游戏画面质感的.如果没有加入光源,游戏场景可能就会显得很昏暗.在unity中提供了三种不同的光源类型:点光源,聚光灯,平行光. 点光源 顾名思义,点光源是从一个点向周围散发出光的光源,就像电灯一样.创建点光源在hierarchy视图中点击create--point light: 创建完以后,点击点光源对象,在右侧inspector视图中可以看到点光源的所有信息: type:光源的类型.有point(点光源),directional

Linux System Programming 学习笔记(九) 内存管理

1. 进程地址空间 Linux中,进程并不是直接操作物理内存地址,而是每个进程关联一个虚拟地址空间 内存页是memory management unit (MMU) 可以管理的最小地址单元 机器的体系结构决定了内存页大小,32位系统通常是 4KB, 64位系统通常是 8KB 内存页分为 valid or invalid: A valid page is associated with an actual page of data,例如RAM或者磁盘上的文件 An invalid page is

【Unity 3D】学习笔记二十七:unity游戏脚本(七)

使用C#编写游戏脚本 在前面提到,unity支持三种语言编写脚本:js,C#,boo.入门的时候建议只用js,因为js比较简单易懂,语法也不是很严格.但后来晋级的时候推荐使用C#,因为它比较符合unity的编程思想,执行效率更高.下面总结下怎么使用C#编写脚本. 继承MonoBehaviour类 在unity中,任何一个脚本,包括上述三种语言都需要去继承MonoBehaviour这个类.为什么我们之前写JS代码的时候没有继承咧?因为在创建JS代码的时候,系统会将其类名与继承关系隐藏起来. 在pr

虚拟机VMWare学习笔记九 - 物理机上的文件挂载到虚拟机上

物理机上的文件夹或盘符直接挂载到虚拟机上使用. VM -- Settings Options -- Shared Folders -- 勾选Always enabled , 勾选Map as a network drive in Windows guests 在点击下面的添加来添加共享的文件夹 选择路径 可以看到虚拟机中的共享文件夹已经出现在Windows 中了 虚拟机VMWare学习笔记九 - 物理机上的文件挂载到虚拟机上

【Unity 3D】学习笔记四十五:游戏实例——击垮围墙

用这个游戏实例来总结之前我们复习的所有关于物理引擎的知识.在游戏中,发送一个球形的炮弹来击垮前面的墙.给炮弹与墙都添加了刚体组件,给炮弹绑定了粒子系统,并且在炮弹的粒子动画中添加了5组不同的颜色,显示移动轨迹. using UnityEngine; using System.Collections; public class Script_06_14 : MonoBehaviour { //炮弹对象 private GameObject obj; //准心贴图 public Texture te