zoukankan      html  css  js  c++  java
  • 辛巴学院-Unity-剑英的c#提高篇(一)主循环

    这是测试版

    辛巴学院:正大光明的不务正业。

    最近刚刚离开了我服务了三年多的公司,因为一个无数次碰到的老问题,没钱了。

    之前不知道做什么好的时候,机缘巧合之下和哒嗒网络的吴总聊了一下,发现了vr game这扇窗户,这里权当帮哒嗒网络打个广告吧。^_^

    回头看看仓惶的这一段时间,荒废了很多,抽空回来再和大家聊聊c#。

    之前做了个入门系列,胡乱说了些东西。感觉入门这样子也就差不多了,该稍微提高一点了。

     

    从写一段程序,到写一个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,未尝不可。然后我们来把他结构化一下吧。

    看看结构化之后的代码

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

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

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

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

     

     

    再会,哈库拉玛塔塔

  • 相关阅读:
    MySQL 可重复读,差点就让我背上了一个 P0 事故
    Thread.sleep(0) 有什么用
    你不会还在用这8个错误的SQL写法吧?
    Spring事务失效的 8 大原因
    我说 SELECT COUNT(*) 会造成全表扫描,面试官让我回去等通知
    这么写参数校验(Validator)就不会被劝退了
    HyperLedger Fabric 1.4 基础环境搭建(7)
    HyperLedger Fabric 1.4 简介(6.1)
    HyperLedger Fabric 1.4 关键技术(6.4)
    HyperLedger Fabric 1.4 交易流程(6.3)
  • 原文地址:https://www.cnblogs.com/crazylights/p/4931312.html
Copyright © 2011-2022 走看看