Unity脚本生命周期与执行顺序

文章目录

  • 脚本生命周期
    • MonoBehavior生命周期图
  • 脚本执行顺序
    • 自定义执行顺序

在Unity中,脚本可以理解为附加在游戏对象上的用于定义游戏对象行为的指令代码。必须绑定在游戏对象上才能开始它的生命周期。游戏对象可以理解为能容纳各种组件的容器,游戏对象的所有组件一起决定了这个对象的行为和游戏中的表现。

脚本生命周期

Unity脚本中的常见必然事件如下表所示

名称 触发时机 用途
Awake 脚本实例被创建时调用 用于游戏对象的初始化,注意Awake的执行早于所有脚本的Start函数
OnEnable 当对象变为可用或激活状态时被调用 用途
Start Update函数第一次运行之前调用 用于游戏对象的初始化
Update 每帧调用一次 用于更新游戏场景和状态
FixedUpdate 每个固定物理时间间隔调用一次 用于物理状态的更新
LateUpdate 每帧调用一次(在update之后调用) 用于更新游戏场景和状态,和相机有关的更新一般放在这里
OnGUI 渲染和处理OnGUI事件 用途
OnDisable 当前对象不可用或非激活状态时被调用 用途
OnDestroy 当前对象被销毁时调用 用途

下面将以代码来看看这些必然事件的调用时机
新建一个C#脚本,并添加以下代码,然后将其挂到任意的游戏对象上

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class TestTest : MonoBehaviour
{
    private void Awake()
    {
        Debug.Log("Awake");
    }

    private void OnEnable()
    {
        Debug.Log("OnEnable");
    }

    // Start is called before the first frame update
    void Start()
    {
        Debug.Log("Start");
    }

    // Update is called once per frame
    void Update()
    {
        Debug.Log("Update");
    }

    private void FixedUpdate()
    {
        Debug.Log("FixedUpdate");
    }

    private void LateUpdate()
    {
        Debug.Log("LateUpdate");
    }

    private void OnGUI()
    {
        Debug.Log("OnGUI");
    }

    private void OnDisable()
    {
        Debug.Log("OnDisable");
    }

    private void OnDestroy()
    {
        Debug.Log("OnDestroy");
    }
}

打印结果如下图示:

可以发现,Awake, Start函数都是在游戏对象被创建时调用一次。
当游戏过程中调整脚本的可见状态时,会分别调用OnEnable, OnDisable函数,而Awake和Start将不会再调用,也就是说一旦脚本被挂载上以后,Awake和Start有且仅会被执行一次。
而Update, FixedUpdate, LateUpdate, OnGUI函数是会在游戏过程中被多次调用的(日志窗口右侧的数字表示该条日志信息打印的次数)。
最后在游戏对象被销毁时,会依次调用OnDisable, OnDestory函数。

MonoBehavior生命周期图

下面再放上一张外国友人所画的生命周期图

脚本执行顺序

在游戏开发中,不可避免的会使用许多脚本,那么如何确定不同脚本之间调用的先后顺序呢

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class Test1 : MonoBehaviour
{
    // Start is called before the first frame update
    void Start()
    {
        Debug.Log("start 1");
    }

    private void Awake()
    {
        Debug.Log("awake 1");
    }

    // Update is called once per frame
    void Update()
    {
        Debug.Log("update 1");
    }
}

将上面的代码依次添加到Test1, Test2和Test3脚本中(适当修改打印的日志),并挂载到不同的游戏对象上。
挂载的顺序为先挂载Test3,再挂载Test2,最后挂载Test1
打印结果如下图所示

打印结果是先打印Test1的,再打印Test2,最后打印Test3。
其实脚本的执行顺序与挂载到游戏对象上的先后顺序有关。最先被挂载的最后执行,最后被挂载的最先执行(如果读者有疑惑,可以不断调整脚本的挂载顺序,看日志打印与上述结论是否相符)。

需要注意的是,无论多个脚本的执行顺序如何,但所有脚本的Awake函数一定会比所有的Start函数先执行完,所有的Start函数一定也会比所有的Update函数先执行完,其他有顺序的生命周期函数也是类似的(从上图的日志信息中也可以看出)。

自定义执行顺序

有时可能有这样的需求,A脚本中的属性实例化可能需要用到B脚本中的属性,所以在A脚本属性实例化时,必须保证B脚本已经被实例化完毕。当然我们可以通过先挂在A脚本再挂载B脚本来实现。但在实际开发中,用到的脚本中多,很难去记住各个脚本挂载的先后顺序。所以Unity提供了Script Execution Order配置项,来配置多个脚本的执行顺序。
在工程面板中任意点击选中一个脚本文件,在属性面板中会出现该脚本的详细信息,选择右上角的Execution Order...,打开如下图所示界面

点击“+”可以添加脚本,为其设置order值,order值越小的越先执行,order值越大的越后执行

原文地址:https://www.cnblogs.com/iwiniwin/p/10793653.html

时间: 2024-10-10 23:09:32

Unity脚本生命周期与执行顺序的相关文章

简单记录一下vue生命周期及 父组件和子组件生命周期钩子执行顺序

首先,vue生命周期可以用下图来简单理解 当然这也是官方文档的图片,详细的vue周期详解请参考这里 然而当同时存在父子组件的时候生命周期钩子是如何执行的呢? 请看下文: 加载渲染过程父beforeCreate->父created->父beforeMount->子beforeCreate->子created->子beforeMount->子mounted->父mounted 子组件更新过程父beforeUpdate->子beforeUpdate->子up

UIViewController的生命周期及执行顺序

当一个视图控制器被创建,并在屏幕上现实的时候.代码的执行顺序: 1.alloc   创建对象,分配空间. 2.init 初始化对象,初始化数据. 3.loadView       从nib载入视图,通常这一步不需要去干涉.除非你没有使用xib文件创建视图 4.viewDidLoad 载入完成,可以进行自定义数据以及动态的创建其他空间. 5.viewWillAppear  视图将出现在屏幕之前. 6.viewDidAppear 视图在屏幕上渲染完成. 当一个视图被移除屏幕并且销毁的时候执行顺序:

unity脚本生命周期

using UnityEngine;using System.Collections; public class CubeScript : MonoBehaviour { //脚本对象加载的时候调用 void Awake(){ Debug.Log ("Awake"); } //脚本可用的时候会被调用一次 void OnEnable(){ Debug.Log ("OnEnable"); } //脚本初始化调用一次 void Start () { Debug.Log(&

Unity 脚本生命周期流程图

渲染 OnPreCull: 在相机剔除场景之前调用此函数.相机可见的对象取决于剔除.OnPreCull 函数调用发生在剔除之前. OnBecameVisible/OnBecameInvisible: 在对象对于相机可见/不可见时调用此函数. OnWillRenderObject: 如果对象可见,则为每个相机调用一次此函数. OnPreRender: 在相机开始渲染场景之前调用此函数. OnRenderObject: 在完成所有常规场景渲染后调用此函数.此时,可使用 GL 类或 Graphics.

Vue 的父组件和子组件生命周期钩子执行顺序

原文地址:https://www.cnblogs.com/anbozhu7/p/11278000.html

(转)unity3d中脚本生命周期(MonoBehaviour lifecycle)

自:http://blog.csdn.net/qitian67/article/details/18516503 最近在做一个小示例,发现类继承于MonoBehaviour的类,有很多个方法,于是乎必然要问出一个问题:这么多个方法,执行先后顺序是如何的呢?内部是如何进行管理的呢?于是在网上找了许多资料,发现了Richard Fine在2012年就已经发布了一篇文章,而且讲得算是相当深入,并且很有道理的,这里加上我的一些尝试与思考,分享给大家. 先贴上图,大家有个直观认识: 接下来,做出一下讲解:

Unity3D热更新之LuaFramework篇[06]--Lua中是怎么实现脚本生命周期的

前言 用c#开发的时候,新建的脚本都默认继承自Monobehaviour, 因此脚本才有了自己的生命周期函数,如Awake,Start, Update, OnDestroy等. 在相应的方法中实现游戏逻辑,引擎会适时调用. 而Lua在这里做为c#的一个外延语言,自然是不受Unity管理的. 不过,在前几篇文章中能看到,在XxxPanel和XxxCtrl中确实又有生命周期的影子,比如在MessagePanel.lua脚本中就存在MessagePanel.Awake和MessagePanel.OnD

Execution Order of Event Functions, unity 3d 事件函数的执行顺序

学习unity3d,感觉事件顺序很重要.就翻译一下官方文档吧. Execution Order of Event Functions 事件函数的执行顺序 In Unity scripting, there are a number of event functions that get executed in a predetermined order as a script executes. This execution order is described below: Unity 脚本中

Servlet的生命周期及执行过程

Servlet生命周期分为实例化.初始化.响应请求调用service()方法.消亡阶段调用destroy()方法. 执行过程如下: 1)当浏览器发送一个请求地址,tomcat会接收这个请求 2)tomcat会读取项目中的web.xml中的配置 3)当请求地址符合servlet-mapping标签映射的地址,会进入这个servlet 4)servlet首先会实例化(构造),然后初始化执行init()方法,init()方法至始至终执行一次,servlet对象是单实例 5)根据请求的方式是get或po