Unity中xLua与toLua对Vector3的优化

Posted 3D游戏引擎

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Unity中xLua与toLua对Vector3的优化相关的知识,希望对你有一定的参考价值。

目录

  1. 概述
    1.1 Vector3的定义
    1.2 主要优化的是什么?

  2. xLua对Vector3的优化
    2.1 xLua创建Vector3
    2.2 xLua获取Vector3 -- C#的Vector3传入lua
    2.3 xLua 设置 Vector3到C#
    2.4 GCOptimize -- PushUnityEngineVector3的由来

  3. toLua对Vector3的优化
    3.1 toLua创建Vector3
    3.2 toLua获取Vector3 -- C#的Vector3传入lua
    3.3 toLua 设置 Vector3到C#

  4. xLua与toLua对Vector3的优化的区别
    4.1 效率性能的比较
    4.2 扩展性的比较

一. 概述

1.1 Vector3的定义

public struct Vector3,是一个struct 结构体,值类型。

1.2 主要优化的是什么?

主要优化 减少gc + 减少lua与C#的交互。

  1. 为什么会产生gc?
    原因是boxing(装箱)和unboxing(拆箱)。Vector3(栈)转为object类型需要boxing(堆内存中),object转回Vector3需要unboxing,使用后释放该object引用,这个堆内存被gc检测到已经没引用,释放该堆内存,产生一个gc内存。

  2. 如何优化gc?
    值拷贝

二. xLua对Vector3的优化

2.1 xLua创建Vector3

lua中有2种方式可以表示Vector3:

  1. 创建Vector3对象,使用userdata:CS.UnityEngine.Vector3(7, 8, 9)

    1. 调用UnityEngineVector3Wrap中函数 static int __CreateInstance(RealStatePtr L)

    2. C#中new一个Vector3:UnityEngine.Vector3 __cl_gen_ret = new UnityEngine.Vector3(x, y, z);

    3. translator.PushUnityEngineVector3(L, __cl_gen_ret);
      要注意的是push方法是PushUnityEngineVector3,普通是translator.Push

      xlua push vec3 to lua 2.png


    4. PushUnityEngineVector3做的优化是申请一块userdata(size=12),将Vector3拆成3个float,Pack到userdata,push到lua

      Unity中xLua与toLua对Vector3的优化

      xlua push float3 to lua 3.png


    5. 这种Vector3 userdata传给C#后,有一个与Pack对应的UnPack 过程。

  2. Table替代 :  {x = 1, y = 2, z = 3}

    • 创建时,不与Unity C#交互(这与toLua类似)

    • 传给C#后,在C# UnPack 这个table,取出x、y、z, 赋值给new Vector3使用。UnPack 在2.3中说明。

2.2 xLua获取Vector3 -- C#的Vector3传入lua

lua中 aTransform.position获取Vector3坐标:

  1. UnityEngineTransformWrap. _g_get_position  lua想从transform获取position

Unity中xLua与toLua对Vector3的优化

transform getposition.png

  1. 获取C# transform对象:UnityEngine.Transform __cl_gen_to_be_invoked = (UnityEngine.Transform)translator.FastGetCSObj(L, 1);

  2. translator.PushUnityEngineVector3(L, __cl_gen_to_be_invoked.position); 这和2.1中创建一个Vector3 push userdata到lua过程一致。

2.3 xLua 设置 Vector3到C#

lua中 aTransform.position = Vector3坐标:

  1. UnityEngineTransformWrap. _s_set_position,  lua想把pos设置到transform.position

    Unity中xLua与toLua对Vector3的优化

    xLua transform setposition.png


  2. 设置position有2.1中的2种方式:

    1. 创建Vector3对象:aTransform.position = CS.UnityEngine.Vector3(7, 8, 9)

    2. Table替代 :  aTransform.position = {x = 1, y = 2, z = 3}
      直接调用 CopyByValue.UnPack,将Table的x,y,z值取出,设置到out UnityEngine.Vector3 val

Unity中xLua与toLua对Vector3的优化

transform setposition get.png

  1. CopyByValue.UnPack
    上面2种方式调用的 CopyByValue.UnPack实现不同;

    1. userdata的方式,Pack的时候,使用xlua_pack_float3,对应的UnPack过程使用xlua_unpack_float3,解出userdata struct.

      Unity中xLua与toLua对Vector3的优化

      xlua userdata方式设置vector3.png

    2. lua table,从栈中依次读取3个float值。


      Unity中xLua与toLua对Vector3的优化

      xlua table方式设置vector3.png

2.4 GCOptimize -- PushUnityEngineVector3的由来

  1. 为何Vector3的push到lua 会有一个针对优化的接口PushUnityEngineVector3
    Vector3 struct配置了GCOptimize属性(对于常用的UnityEngine的几个struct,Vector系列,Quaternion,Color。。。均已经配置了该属性),这个属性可以通过配置文件或者C# Attribute实现;

    Unity中xLua与toLua对Vector3的优化

    GCOptimize Vector3.png


  2. 从GCOptimize列表中去掉Vector3 会怎么样呢?
    PushUnityEngineVector3接口就不存在了,而Vector3的push到lua会使用translator.Push(L, __cl_gen_ret); ,不做优化public void Push(RealStatePtr L, object o),会产生boxing(装箱)和unboxing(拆箱),代表着一个gc。

    Unity中xLua与toLua对Vector3的优化

    去掉vector3的GCOptimize后的push.png



三. toLua对Vector3的优化

3.1 toLua创建Vector3

Vector3.New(x, y, z)

toLua并没有跟Unity C#交互.

3.2 toLua获取Vector3 -- C#的Vector3传入lua

lua中 aTransform.position获取Vector3坐标:

  1. C#  UnityEngine_TransformWrap.get_position;调用ToLua.Push

    Unity中xLua与toLua对Vector3的优化

    toLua get_position.png


  2. C#传入Vector3的x,y,z;


    Unity中xLua与toLua对Vector3的优化

    toLua get_position pushvec3.png

  3. 在lua建一个lua table,把x,y,z设置为对应字段;

  4. 设置该table的metatable为Vector3.lua的方法实现;

3.3 toLua 设置 Vector3到C#

  1. 从栈中取出对应table的x,y,z字段


    toLua set_position ToVector3.png

  2. C# new一个Vector3,将x,y,z赋值到Vector3;
    与xLua的table替代方式非常类似。

四. xLua与toLua对Vector3的优化的区别

  1. 效率性能的比较,toLua高

    1. xLua与toLua都不产生gc

    2. xLua在创建Vector3的userdata方式和Vector3的方法调用,都需要跟Unity C#交互;而toLua在这两方面是纯Lua端执行,无需跟Unity C#交互,效率最高。

    3. xLua有一个特点:所有无GC的类型,它的数组访问也没有GC。

  2. 扩展性的比较,xLua高
    toLua重新Lua实现的类,需要增加一种新的值类型十分困难, 数量有限,并且与Unity C# Vector3核心代码深度耦合。

toLua lua类列表.png


xLua支持的struct类型宽泛的多,包含枚举,用户要做的事情也很简单,用GCOptimize声明一下类型即可。支持自定义struct。(struct要求1.含无参构造函数  2.只包含值类型,可以嵌套其它只包含值类型的struct)

相关链接

  1. xlua github

    https://github.com/Tencent/xLua

  2. xlua特性

    https://github.com/Tencent/xLua/blob/master/Assets/XLua/Doc/features.md

  3. tolua 官网

    http://www.ulua.org/index.html

  4. tolua github

    https://github.com/topameng/tolua

  5. Unity下XLua方案的各值类型GC优化深度剖析

        https://www.gameres.com/700911.html

以上是关于Unity中xLua与toLua对Vector3的优化的主要内容,如果未能解决你的问题,请参考以下文章

xlua中hotfix简单实用

热更新--lua

Unity3D游戏轻量级xlua热修复框架

Unity3D游戏xlua轻量级热修复框架

Xlua对c#的vector3等结构体的优化

Unity 游戏用XLua的HotFix实现热更原理揭秘