简介

  在基于lua进行热更新的项目中,我们通常会通过luaBehaviour来让lua文件模拟MonoBehaviour,可以让lua文件拥有一些MonoBehaviour的生命周期,如Enable、Disable、Update。

  同时可以注入一些UnityEngine.Object。在lua中方便的调用Unity中的对象。方便开发者进行开发。

  本文进行介绍的luaBehaviour就是基于这个思路设计的,除了上面提到的特性之外还通过Json支持了更多类型的注入,Editor界面更人性化的展示,整个开发过程更接近MonoBehaviour的开发体验。

使用

1.新建继承LuaBehaviour的lua文件,通过AddDefineList添加需要序列化的信息。



2.直接把这个lua文件拖到CS中的LuaBehaviour上即可。





  接下来可以像MonoBehaviour一样对数据进行编辑了。

实现

脚本的序列化

  luaBehaviour在使用的时候一般是通过记录名字或路径的方式来序列化lua文件的的。打包后通过这个路径从AssetBundle中进行lua文件的加载(这个是lua脚本热更新常用的策略,在这边就不多说了)。

  这里还是通过路径来记录lua文件,但是不需要手动输入,而是通过直接拖动的方式间接记录lua文件路径。

注入的方式

  不通过C#侧定义注册信息,而是通过在Lua中先定义好要注册的类型和名称(更符合MonoBehaviour的开发姿势)。开发时,Editor读取lua中定义的类型和名称信息,展示在Inspector中,供开发者编辑。

  在运行时再将序列化的数据注入到lua实例中。

序列化的实现

UnityEngine.Object类型对象的序列化

  这种类型没啥特别的,在LuaBehaviour脚本中定义一个记录key和UnityEngine.Object的List即可。

其他类型的序列化

  非UnityEngine.Object类型就没有一个统一的格式,无法通过一个列表简单的进行记录。但是我们查看Prefab的实例可以发现,Prefab序列化的数据其实和Json很像。对在MonoBehaviour中定义的各个字段的序列化姿势也和Json很像。



  所以这里考虑使用Json来序列化非UnityEngine.Object类型的对象。刚好Unity由提供了一套简单高效的Json工具JsonUtility。JsonUtility内部就是通过Unity serializer实现的,所以稳定性很有保证。

  但是JsonUtility有个缺点,只能序列化一部分类型,不能序列化如Int、List这种类型。

  为了解决这个问题。这里在序列化的时候通过泛型为每个类型生成一个Wrap类型。即可通用的实现各种类型的序列化。

注入的实现

UnityEngine.Object类型对象的注入

  也没啥好说的,直接根据key向Lua实例中Set即可。

其他类型的注入

  先实例化为Wrap对象,然后再取出其中的需要注入的对象,Set到Lua对象中即可。

Inspector界面中的展示

UnityEngine.Object类型对象的展示

  UnityEngine.Object类型统一使用EditorGUILayout.ObjectField绘制即可。

其他类型的展示

  因为没有找到一个通用的可以表现所有对象的绘制方式,所以这里也做了一个转换。先通过Emit生成一个继承自ScriptableObject的类(因为ScriptableObject是UntiyEngine.Object,所以可以使用SerializedObject来绘制。同时可以直接通过ScriptableObject.CreateInstance进行实例的创建)。把需要绘制的对象放到这个类里面,然后通过EditorGUILayout.PropertyField绘制即可。

Enable、Update等函数的调用

  这里把Update、FixedUpdate等高频或者很少使用的函数拆分出去,只有在Lua中定义了这些函数,才添加对应的Assistant脚本对这些函数进行调用。



  Enable、Disable、Destroy这三个常用函数,就直接放在LuaBehaviour脚本中进行调用。

Tips

  1. 这篇文章只起到大致思路和关键点的说明,具体细节可以直接看代码,代码比较少也比较清晰。
  2. 通过Wrap的方式序列化各种对象的方式其实也可以考虑用到一些用户数据在客户端的持久化。
  3. 因为这边主要展示LuaBehanviour的功能,所以AssetBundle的生成和从AssetBundle中加载lua文件都写得很临时,仅作展示用。
  4. 打包测试之前要用AssetBundles->Build AssetBundle For Lua生成一下bundle。

项目链接:https://github.com/blueberryzzz/LuaBehaviour

Xlua中LuaBehaviour的实现的更多相关文章

  1. xLua中导出Dotween

    前言 在xlua的lua脚本中使用dotween,官方的文档中有提到可以导出,但未介绍详细的步骤,相信比较多的朋友有需要,刚好项目中也在使用xlua和dotween,所以做个笔记. 基础知识: xLu ...

  2. 【第二篇】xLua中lua加载方式

     xLua中lua文件加载方式 1. 直接执行字符串方式 LuaEnv luaenv = new LuaEnv(); luaenv.DoString("CS.UnityEngine.Debu ...

  3. xLua中C#调用Lua

    C#调用Lua 一.前提 这里使用的是XLua框架,需要提前配置xlua,设置加载器路径: 可以参考之前的Blog:<xlua入门基础>: 二.C#调用Lua全局变量 lua中所有的全局变 ...

  4. xLua中Lua调用C#

    xLua中Lua调用C# 1.前提 这里使用的是XLua框架,需要提前配置xlua,设置加载器路径: 可以参考之前的Blog:<xlua入门基础>: //调用段,所有的lua代码都写在Lu ...

  5. xlua中lua对象到c#对象的转型

    lua中的类型 基础类型 #define LUA_TNIL 0 #define LUA_TBOOLEAN 1 #define LUA_TLIGHTUSERDATA 2 #define LUA_TNUM ...

  6. 现学现卖】IntelliJ+EmmyLua 开发调试Unity中Xlua

    http://blog.csdn.net/u010019717/article/details/77510066?ref=myread http://blog.csdn.NET/u010019717 ...

  7. unity+xlua开发中的问题笔记

    转载请标明出处:http://www.cnblogs.com/zblade/ 一.概述 整理遇到的一些较难处理的bug,总结相关经验 二.主要问题 2.1 material类型的依赖修改 对于mate ...

  8. xlua 实现协程替换Unity中的协程

    C#中的协程: IEnumerator ShowSpiritInfo() { UIMessageMgr.ShowMsgWait(true); DestroyUIModelInfo(); bool is ...

  9. xlua怎么样hotfix C#中的重写方法???

    问题的来源之这样的: 线上项目遇到一个问题,就是子类 override 了父类的一个 virtual 方法,并且调用到了父类里面的 virtual 方法.现在子类  override 的方法里有一些错 ...

随机推荐

  1. django.core.exceptions.ImproperlyConfigured: mysqlclient 1.3.13 or newer is required; you have 0.9.3.解决办法

    "E:\API_Manager_PlatForm\venv\lib\site-packages\django\db\backends\mysql\base.py"在这个路径里件把b ...

  2. less : 解决升级后报错的问题

    vue2项目. 上版本. { "name": "xxx", "version": "1.0.0", "desc ...

  3. 前端练习错题day01

    <1>.css盒子模型中,padding是透明的,这一部分可以显示背景. <2>.注意&&符号左右条件先后顺序,可能会报错. <3>.在 HTML5 ...

  4. 【mysql】- 索引使用篇

    回顾 每个索引都对应一棵B+树,B+树分为好多层,最下边一层是叶子节点,其余的是内节点.所有用户记录都存储在B+树的叶子节点,所有目录项记录都存储在内节点. InnoDB 存储引擎会自动为主键(如果没 ...

  5. Java Web(2)-jQuery上

    一.jQuery初体验 使用jQuery给一个按钮绑定单击事件 <!DOCTYPE html> <html lang="en"> <head> ...

  6. 数字麦克风PDM信号采集与STM32 I2S接口应用(四)--单片机源码

    本文是数字麦克风笔记文章的单片机程序.一些朋友私信我,调试出问题. 我就把源码贴出来吧,可能主要问题是DMA的配置. 尤其双DMA时候,需要手动启动I2S的接收DMA,HAL库没有这个接口,不看dat ...

  7. Nginx(一)Linux上的Nginx安装步骤

    一.Windows下安装 解压:nginx-windows 双击: nginx.exe 能看到nginx欢迎界面说明,nginx安装成功 演示下 nginx做静态服务器 二.Linux下安装 (1). ...

  8. 我自己总结的sqlite的命令行命令集

    我自己总结的sqlite 的命令行命令 导入文本数据文件时,设置分隔符为","sql>.separator "," sql>import devic ...

  9. Lun4R-CyBRICSCTF wp

    WEB Hunt (Web, Baby, 50 pts) 打断点,然后就一个一个被抓住了... 接着F12就出现了.(这个flag是白色的,藏在下面....)... RE Baby Rev 题目给了个 ...

  10. python beautifulsoup基本用法-文档结构

    一.BeautifulSoup概述 BeautifulSoup是python的一个库,用于接收一个HTML或XML字符串并对其进行格式化,然后使用提供的方法快速查找指定元素. 使用BeautifulS ...