从写一段程序,到写一个app,写一个游戏,到底其中有什么不同呢?一段程序的执行时间很短,一个应用的执行时间很长,仅此而已。

游戏中存在一个帧的概念。

 

这个概念大家都知道,类比的话,它就是电影胶卷的格。一格出现一段时间,然后换下一格。

电影一秒24格,游戏用的是一秒30帧,60帧。

电影胶片是在镜头前一格一格的放映,游戏逻辑是在update中一帧一帧的执行。

那么Update函数的驱动是如何完成的呢?

这个东西就是主循环

先来看看一个基本的控制台程序

 

唔,helloworld,这个程序我们已经很熟悉了。一闪而过的那个版本。因为他执行然后马上就结束了。

那么这个版本呢?

 

你会说,我去,你写了一个死循环。

是的,主循环就是一个死循环。有了这个死循环,一段程序就能逐渐成为一个应用,一个游戏。

 

让我们把update拆出来,是不是看起来有点熟悉了。

 

再来个OnStart,是不是更加熟悉了

 

原来unity的MonoBehaviour就是这样啊。

任何的程序中都存在着主循环,在常用的界面框架中,通常隐藏主循环,留下事件型的接口。

主循环很简单吧,仅仅是看起来这样。

游戏程序通常事件型的框架不多,大部分逻辑需要从主循环层次开始组建,也就是你需要对主循环的理解非常深,能够从主循环上创建很多种模式的逻辑。

我们开始来介绍几个常见的

主循环和定时器

试想如下需求,需要每三秒钟打印一条log,怎么做这个三秒钟的计时?

电影是24格,每秒钟24帧,稳定的,雷打不动。通过数帧就知道个大概。

可是大家都知道游戏的帧率是跳跃的,每一帧的时间不固定。

Unity提供了一个参数,告诉你上一帧开始,到这一帧开始经过的时间,单位为秒 Time.Deltatime;

 

因为每一帧开始到上一帧开始的间隔给你了。只要把他们累加起来,就是一个计时器。你可以用秒表确认一下这个程序,他的表现和你的秒表一样精准,分毫不差。

这就是一个计时器了,一个用不稳定的帧速率的主循环驱动的计时器。只是他光计时了,啥也没干,我们来让他做点什么。

 

这个计时器是最基本的一个逻辑,固定帧率在游戏中只是一个理想化状态,大部分情况下无法实现固定帧率。

几乎所有的游戏逻辑都涉及到计时问题,都会涉及到如何在浮动帧率时进行逻辑控制。

时间是连续累加的,请时刻记住这一点。

主循环与缓动

用定时器去驱动数值的变化,在一些情况下,被称为缓动,dotween,itween,名字里有个tween的这类库或者插件,都是做缓动的。

他们包装了各式各样的缓动的模式代码,帮你节约一点时间。

我们来让一个box 3秒钟从A点移动到B点,所有缓动的系统中都采用这样一个考虑方式,零为开始,一为结束。那么我们让我们的timer 几秒完成一个从零到一的过程,这就是一个缓动的周期。

就用我们之前的代码去考虑,我们的timer 加到3秒,然后变小,再到三,再变小,ok。直接给timer+=的时候乘一个系数可以改变timer变化的速度,可以调节为三秒钟完成从零到一,但是我们入门篇讲过,好代码从命名开始,既然他叫做timer,他就不应该从零到一,从零到一的标识的是缓动的进度。

取名为lerp

 

如上代码就完成了一个缓动逻辑,把这个脚本给一个cube,然后begin end 填入不一样的值,跑起来看看。

无论你用了什么高大上的缓动库,知其然总是有好处的,而这,就是缓动框架背后的事情。

主循环和状态机

入门篇我们就讲了一个时空观的问题,这个用程序的概念就是状态,图灵机、冯诺依曼机,是计算机的基础,这个基础说的就是状态机。

即使从更大的尺度来看,高级语言来看。在顺序执行的程序体系中,状态机依然是编程的基础。

我们前面告诉过大家单步调试,每一个断点,就是一个状态。

程序是由一个一个的状态构成的。

再从更大的尺度看,功能模块和程序结构的角度,状态机依然是功能的基础,我现在处于主菜单还是战斗菜单?我现在是在充值还是还消费?

同级别的功能某一时刻必然处于某一功能中,是我们的设计基础。

游戏框架从哪里开始,必然从状态的分割开始。有些框架提供了一个比状态机更高级一点的模式,导航器,其实也属于状态机,但是导航器记录着之前的状态,可以NavBack。

导航器模式是什么?随便点开一个手机app,点一个功能,再点一个功能,然后按back,back,这就是导航器。

现在由于手游流行起来,游戏界面效法了很多app的设计,所以现在导航器设计是一个非常主流的设计。

由于我们只是一个抛砖引玉的提高过程,我们这里不会去写一个导航器框架出来,我们只是去解释状态机背后的行为。

这个状态代码这么写,有点长,我们分两段来看,一段是update函数

 

一段是ongui函数

 

仔细看过这段代码,你也许会说,你骗人,这是个锤子的状态机哦,这就是个if else

这就是个if else,状态机的本质就是switch case,就是 if else,除非,给他一个结构化的设计。

当你的逻辑状态只有两三个的时候,if else,未尝不可。然后我们来把他结构化一下吧。

看看结构化之后的代码

 

我们抽象出一个表达状态的接口,于是主要的代码变得非常简洁,这就是一个状态机咯。

不过实现代码就不那么简洁了

 

有这样的设计作为基础,添加再多的状态也不怕啦,不过这也只是一个解释用的代码。

实际操作时肯定比这个面对的问题要更复杂一些。

题外话:JAVA语言,JAVA字节码,JAVA运行时,JAVA基本类库,都叫java。

而微软这边分别叫C#语言,MSIL字节码,dotnet运行时,dotnet framework。其实就是为了怕被告。

最新文章

  1. iOS报错[__NSCFNumber length]: unrecognized
  2. 使用 {$INCLUDE} 或 {$I} 指令管理和调用自定义函数
  3. Hibernate + proxool 连接数超过最大允许连接数
  4. 例题.点击按钮显示内容+弹窗效果+ajax
  5. 算法起步之Bellman-Ford算法
  6. 从一篇ICLR'2017被拒论文谈起:行走在GAN的Latent Space
  7. C++中 Rand随机序列函数
  8. 并发编程之ThreadLocal、Volatile、synchronized、Atomic关键字扫盲
  9. swiper 逆向轮播
  10. 免费为网站加上HTTPS
  11. Linux基础实操五
  12. javaweb数据库编程代码详细讲解
  13. 检查文件是否被修改或者被破坏工具 md5
  14. Android原生(Native)C开发之四:SDL移植笔记
  15. php后台对接ios,安卓,API接口设计和实践完全攻略,涨薪必备技能
  16. 安卓——implements——OnClickListener
  17. django之block extend标签
  18. jQuery(六):value值操作
  19. Spring+Swagger文档无法排序问题解决
  20. Linux Terminator

热门文章

  1. <c和指针>学习笔记3之函数和数组
  2. idea+tomcat 解决 debug超级慢 问题
  3. 转: Charles 从入门到精通
  4. UVa 11552 Fewest Flops (DP)
  5. surface shader相关参数,命令
  6. poj3164(最小树形图&朱刘算法模板)
  7. 51nod1412(dp)
  8. axios发送两次请求原因及解决方法
  9. 【实验吧】该题不简单——writeup
  10. JSF 与 struts2