xLua 内存管理的简单理解

本贴最后更新于 581 天前,其中的信息可能已经事过境迁

xLua 是目前 Unity 热更新中用的比较多的方案之一,花了些时间看了下代码,并对其中的 Lua 和 C#中对象的传递与内存管理有了一些个人理解,可能理解有误,欢迎批评指正

C#对象传递到 Lua

Unity 中 C#对象可以大致分成几类:继承自 primitive 类型、string 类型、struct 类型、enum 类型和类对象,这几种类型的对象有不同的处理逻辑,分为如何将对象传递到 lua 中,当从 lua 中传递回 C#时如何找到对应的 C#对象,以及何时释放对象的引用避免内存泄漏几个方面

  1. primitive 类型
    primitive 类型不参与 GC,所以直接调用 lua_pushxxx 函数,Lua 传递到 C#时也是直接调用 lua_toxxx 即可

  2. string 类型
    string 类型和 primitive 一样,直接调用 lua_pushstring 函数,Lua 传递到 C#时直接调用 lua_tostring 即可

  3. 类对象
    类对象会参与 GC,会把 Lua 查找 C#对象的映射关系存储到 objects 对象中,映射关系存储到 reverseMap 中,这样后续可以直接复用之前的对象

    因为回存在对象回收的问题,所以会对在 Lua 中创建的相应关系的 LuaTable 注册名为 __gc 的 metatable,这样在 Lua 对象被 GC 的时候,可以完全清除掉 objects 对象中对 C#对象的引用,避免内存泄漏,同时 objects 对象中也是一个会扩容的对象池,清除掉引用之后就也可以复用直接的 id,避免无限扩容

    对于继承自 UnityEngine.Object 的对象因为重写了 == 运算符还会有特殊逻辑,需要使用者在合适的时机(一般来讲是 Update 方法中间隔固定时间)调用 LuaEnv 上的 Tick 方法,这样可以在 UnityEngine.Object 对象被销毁后及时将 objects 中 Lua 查找 C#的映射关系单向清理掉,避免内存泄漏,反向的映射关系还是需要等到上面提到的 __gc 方法中清除,避免使用者将 lua 对象再次传回 C#中后因为复用查找到不同的对象出现错误

  4. enum 类型
    枚举类型是固定不变的,不需要参与 GC,会把通过 Lua 查找 C#的映射关系存储到 objects 对象中,C#到 Lua 的映射关系存储到 enumMap 对象中,这样在后续的使用中都会直接复用之前的对象

  5. struct 类型
    因为 c#中 struct 为值类型,所以会出现当一个 struct 对象传递到 Lua 后再传递回 C#就不会是同一个对象了,如果只是使用例如 Color 或者 Vector 等 struct 是没有问题的,但是对于一些内部带有状态的 struct 是会出现问题的,这点需要额外注意

Lua 对象传递到 C#

Lua 中常用的会传递到 C#中的类型有几种:boolean、number、string、table、function,其中 boolean、number、string 都是用是直接值传递方式

  1. function
    function 一般来讲是对应 C#中的 delegate,但是因为 lua 为弱类型,C#为强类型,所以需要一个比较复杂的包装处理,首先会包装成 DelegateBridge 对象,然后使用 getDelegate 函数根据需要的 delegate 类型做一层转换包装处理,返回 DelegateBridge 对象上一个具体的需要的类型的成员函数,并将 lua 对象在 C#中的映射关系存储到 delegate_bridges 中,这样在下次将这个函数从 lua 中传递到 C#中时可以直接复用已有对象
    此外,还有个类叫做 LuaFunction 也可以实现用于将 lua 对象传递到 C#中使用的功能
  2. table
    table 的情况比较复杂,常见的分成几种情况:
    1. table 转换为 C#中 struct,这种情况就是直接构造了一个 struct 对象然后队成员变量赋值
    2. table 转换为 interface,这种情况需要 xLua 生成一个继承自对应 interface 的一个实现类,并对里面所有的抽象方法利用 Lua 的 C API 生成对应的具体实现
    3. 直接将 table 封装为 LuaTable 对象,等到具体使用的时候直接调用 LuaTable 对应方法获取上面的函数或变量

DelegateBridge、实现 interface、LuaTable 等方式都需要注意,他们都会对对应的 table 产生一次引用,而且他们(包括其他类似类也是一样)都继承自 LuaBase,会在析构函数中解除对 table 的引用,避免内存泄漏

  • Unity

    Unity 是由 Unity Technologies 开发的一个让开发者可以轻松创建诸如 2D、3D 多平台的综合型游戏开发工具,是一个全面整合的专业游戏引擎。

    25 引用 • 7 回帖 • 140 关注
  • xLua
    1 引用
1 操作
jerryhwq 在 2023-08-10 16:22:51 更新了该帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • Ngui

    Ngui 是一个 GUI 的排版显示引擎和跨平台的 GUI 应用程序开发框架,基于
    Node.js / OpenGL。目标是在此基础上开发 GUI 应用程序可拥有开发 WEB 应用般简单与速度同时兼顾 Native 应用程序的性能与体验。

    7 引用 • 9 回帖 • 397 关注
  • 30Seconds

    📙 前端知识精选集,包含 HTML、CSS、JavaScript、React、Node、安全等方面,每天仅需 30 秒。

    • 精选常见面试题,帮助您准备下一次面试
    • 精选常见交互,帮助您拥有简洁酷炫的站点
    • 精选有用的 React 片段,帮助你获取最佳实践
    • 精选常见代码集,帮助您提高打码效率
    • 整理前端界的最新资讯,邀您一同探索新世界
    488 引用 • 384 回帖 • 3 关注
  • JVM

    JVM(Java Virtual Machine)Java 虚拟机是一个微型操作系统,有自己的硬件构架体系,还有相应的指令系统。能够识别 Java 独特的 .class 文件(字节码),能够将这些文件中的信息读取出来,使得 Java 程序只需要生成 Java 虚拟机上的字节码后就能在不同操作系统平台上进行运行。

    180 引用 • 120 回帖
  • Kubernetes

    Kubernetes 是 Google 开源的一个容器编排引擎,它支持自动化部署、大规模可伸缩、应用容器化管理。

    116 引用 • 54 回帖
  • 浅吟主题

    Jeffrey Chen 制作的思源笔记主题,项目仓库:https://github.com/TCOTC/Whisper

    1 引用 • 28 回帖
  • Anytype
    3 引用 • 31 回帖 • 14 关注
  • PHP

    PHP(Hypertext Preprocessor)是一种开源脚本语言。语法吸收了 C 语言、 Java 和 Perl 的特点,主要适用于 Web 开发领域,据说是世界上最好的编程语言。

    179 引用 • 408 回帖 • 486 关注
  • Vim

    Vim 是类 UNIX 系统文本编辑器 Vi 的加强版本,加入了更多特性来帮助编辑源代码。Vim 的部分增强功能包括文件比较(vimdiff)、语法高亮、全面的帮助系统、本地脚本(Vimscript)和便于选择的可视化模式。

    29 引用 • 66 回帖
  • Lute

    Lute 是一款结构化的 Markdown 引擎,支持 Go 和 JavaScript。

    27 引用 • 196 回帖 • 27 关注
  • 反馈

    Communication channel for makers and users.

    124 引用 • 928 回帖 • 267 关注
  • 数据库

    据说 99% 的性能瓶颈都在数据库。

    345 引用 • 724 回帖 • 1 关注
  • 工具

    子曰:“工欲善其事,必先利其器。”

    295 引用 • 749 回帖
  • abitmean

    有点意思就行了

    29 关注
  • Wide

    Wide 是一款基于 Web 的 Go 语言 IDE。通过浏览器就可以进行 Go 开发,并有代码自动完成、查看表达式、编译反馈、Lint、实时结果输出等功能。

    欢迎访问我们运维的实例: https://wide.b3log.org

    30 引用 • 218 回帖 • 640 关注
  • 一些有用的避坑指南。

    69 引用 • 93 回帖
  • Rust

    Rust 是一门赋予每个人构建可靠且高效软件能力的语言。Rust 由 Mozilla 开发,最早发布于 2014 年 9 月。

    58 引用 • 22 回帖 • 2 关注
  • 黑曜石

    黑曜石是一款强大的知识库工具,支持本地 Markdown 文件编辑,支持双向链接和关系图。

    A second brain, for you, forever.

    21 引用 • 196 回帖
  • 域名

    域名(Domain Name),简称域名、网域,是由一串用点分隔的名字组成的 Internet 上某一台计算机或计算机组的名称,用于在数据传输时标识计算机的电子方位(有时也指地理位置)。

    43 引用 • 208 回帖 • 1 关注
  • 阿里云

    阿里云是阿里巴巴集团旗下公司,是全球领先的云计算及人工智能科技公司。提供云服务器、云数据库、云安全等云计算服务,以及大数据、人工智能服务、精准定制基于场景的行业解决方案。

    84 引用 • 324 回帖
  • 笔记

    好记性不如烂笔头。

    311 引用 • 796 回帖
  • 房星科技

    房星网,我们不和没有钱的程序员谈理想,我们要让程序员又有理想又有钱。我们有雄厚的房地产行业线下资源,遍布昆明全城的 100 家门店、四千地产经纪人是我们坚实的后盾。

    6 引用 • 141 回帖 • 588 关注
  • API

    应用程序编程接口(Application Programming Interface)是一些预先定义的函数,目的是提供应用程序与开发人员基于某软件或硬件得以访问一组例程的能力,而又无需访问源码,或理解内部工作机制的细节。

    78 引用 • 430 回帖
  • Postman

    Postman 是一款简单好用的 HTTP API 调试工具。

    4 引用 • 3 回帖 • 4 关注
  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3194 引用 • 8214 回帖
  • App

    App(应用程序,Application 的缩写)一般指手机软件。

    91 引用 • 384 回帖
  • Swagger

    Swagger 是一款非常流行的 API 开发工具,它遵循 OpenAPI Specification(这是一种通用的、和编程语言无关的 API 描述规范)。Swagger 贯穿整个 API 生命周期,如 API 的设计、编写文档、测试和部署。

    26 引用 • 35 回帖 • 1 关注
  • TextBundle

    TextBundle 文件格式旨在应用程序之间交换 Markdown 或 Fountain 之类的纯文本文件时,提供更无缝的用户体验。

    1 引用 • 2 回帖 • 73 关注