xLua 内存管理的简单理解

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

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 回帖 • 121 关注
  • xLua
    1 引用
1 操作
jerryhwq 在 2023-08-10 16:22:51 更新了该帖

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • SEO

    发布对别人有帮助的原创内容是最好的 SEO 方式。

    35 引用 • 200 回帖 • 31 关注
  • Vim

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

    29 引用 • 66 回帖
  • Facebook

    Facebook 是一个联系朋友的社交工具。大家可以通过它和朋友、同事、同学以及周围的人保持互动交流,分享无限上传的图片,发布链接和视频,更可以增进对朋友的了解。

    4 引用 • 15 回帖 • 449 关注
  • 设计模式

    设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结出来的。

    201 引用 • 120 回帖
  • Q&A

    提问之前请先看《提问的智慧》,好的问题比好的答案更有价值。

    9894 引用 • 44980 回帖 • 80 关注
  • Kafka

    Kafka 是一种高吞吐量的分布式发布订阅消息系统,它可以处理消费者规模的网站中的所有动作流数据。 这种动作(网页浏览,搜索和其他用户的行动)是现代系统中许多功能的基础。 这些数据通常是由于吞吐量的要求而通过处理日志和日志聚合来解决。

    36 引用 • 35 回帖 • 5 关注
  • LaTeX

    LaTeX(音译“拉泰赫”)是一种基于 ΤΕΧ 的排版系统,由美国计算机学家莱斯利·兰伯特(Leslie Lamport)在 20 世纪 80 年代初期开发,利用这种格式,即使使用者没有排版和程序设计的知识也可以充分发挥由 TeX 所提供的强大功能,能在几天,甚至几小时内生成很多具有书籍质量的印刷品。对于生成复杂表格和数学公式,这一点表现得尤为突出。因此它非常适用于生成高印刷质量的科技和数学类文档。

    12 引用 • 59 回帖 • 5 关注
  • Docker

    Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的操作系统上。容器完全使用沙箱机制,几乎没有性能开销,可以很容易地在机器和数据中心中运行。

    496 引用 • 934 回帖
  • jsoup

    jsoup 是一款 Java 的 HTML 解析器,可直接解析某个 URL 地址、HTML 文本内容。它提供了一套非常省力的 API,可通过 DOM,CSS 以及类似于 jQuery 的操作方法来取出和操作数据。

    6 引用 • 1 回帖 • 487 关注
  • 数据库

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

    345 引用 • 751 回帖 • 1 关注
  • Maven

    Maven 是基于项目对象模型(POM)、通过一小段描述信息来管理项目的构建、报告和文档的软件项目管理工具。

    188 引用 • 319 回帖 • 248 关注
  • RESTful

    一种软件架构设计风格而不是标准,提供了一组设计原则和约束条件,主要用于客户端和服务器交互类的软件。基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。

    30 引用 • 114 回帖
  • Spring

    Spring 是一个开源框架,是于 2003 年兴起的一个轻量级的 Java 开发框架,由 Rod Johnson 在其著作《Expert One-On-One J2EE Development and Design》中阐述的部分理念和原型衍生而来。它是为了解决企业应用开发的复杂性而创建的。框架的主要优势之一就是其分层架构,分层架构允许使用者选择使用哪一个组件,同时为 JavaEE 应用程序开发提供集成的框架。

    947 引用 • 1460 回帖
  • webpack

    webpack 是一个用于前端开发的模块加载器和打包工具,它能把各种资源,例如 JS、CSS(less/sass)、图片等都作为模块来使用和处理。

    42 引用 • 130 回帖 • 250 关注
  • 微信

    腾讯公司 2011 年 1 月 21 日推出的一款手机通讯软件。用户可以通过摇一摇、搜索号码、扫描二维码等添加好友和关注公众平台,同时可以将自己看到的精彩内容分享到微信朋友圈。

    133 引用 • 796 回帖 • 1 关注
  • Unity

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

    25 引用 • 7 回帖 • 121 关注
  • WiFiDog

    WiFiDog 是一套开源的无线热点认证管理工具,主要功能包括:位置相关的内容递送;用户认证和授权;集中式网络监控。

    1 引用 • 7 回帖 • 611 关注
  • SpaceVim

    SpaceVim 是一个社区驱动的模块化 vim/neovim 配置集合,以模块的方式组织管理插件以
    及相关配置,为不同的语言开发量身定制了相关的开发模块,该模块提供代码自动补全,
    语法检查、格式化、调试、REPL 等特性。用户仅需载入相关语言的模块即可得到一个开箱
    即用的 Vim-IDE。

    3 引用 • 31 回帖 • 111 关注
  • OkHttp

    OkHttp 是一款 HTTP & HTTP/2 客户端库,专为 Android 和 Java 应用打造。

    16 引用 • 6 回帖 • 92 关注
  • 黑曜石

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

    A second brain, for you, forever.

    24 引用 • 242 回帖
  • Wide

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

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

    30 引用 • 218 回帖 • 635 关注
  • Python

    Python 是一种面向对象、直译式电脑编程语言,具有近二十年的发展历史,成熟且稳定。它包含了一组完善而且容易理解的标准库,能够轻松完成很多常见的任务。它的语法简捷和清晰,尽量使用无异义的英语单词,与其它大多数程序设计语言使用大括号不一样,它使用缩进来定义语句块。

    554 引用 • 675 回帖
  • Solidity

    Solidity 是一种智能合约高级语言,运行在 [以太坊] 虚拟机(EVM)之上。它的语法接近于 JavaScript,是一种面向对象的语言。

    3 引用 • 18 回帖 • 433 关注
  • 叶归
    9 引用 • 43 回帖 • 20 关注
  • SSL

    SSL(Secure Sockets Layer 安全套接层),及其继任者传输层安全(Transport Layer Security,TLS)是为网络通信提供安全及数据完整性的一种安全协议。TLS 与 SSL 在传输层对网络连接进行加密。

    70 引用 • 193 回帖 • 414 关注
  • Windows

    Microsoft Windows 是美国微软公司研发的一套操作系统,它问世于 1985 年,起初仅仅是 Microsoft-DOS 模拟环境,后续的系统版本由于微软不断的更新升级,不但易用,也慢慢的成为家家户户人们最喜爱的操作系统。

    227 引用 • 476 回帖
  • 倾城之链
    23 引用 • 66 回帖 • 169 关注