Unity中 Xlua使用整理(二)

1.Xlua的配置应用

xLua所有的配置都支持三种方式:打标签静态列表动态列表
配置要求
列表方式均必须是static的字段/属性
列表方式均必须放到一个static类
建议不用标签方式
建议列表方式配置放Editor目录(如果是Hotfix配置,而且类位于Assembly-CSharp.dll之外的其它dll,必须放Editor目录)

1.打标签

xLua用白名单来指明生成哪些代码,而白名单通过attribute来配置,该方式方便,但在il2cpp下会增加不少的代码量,不建议使用

1.LuaCallCSharp

一个C#类型加了这个配置,xLua会生成这个类型的适配代码(包括构造该类型实例,访问其成员属性、方法,静态属性、方法)。一个类型的扩展方法(Extension Methods)加了这配置,也会生成适配代码并追加到被扩展类型的成员方法上。xLua只会生成加了该配置的类型,不会自动生成其父类的适配代码,当访问子类对象的父类方法,如果该父类加了LuaCallCSharp配置,则执行父类的适配代码。如果没有适配代码则会尝试用反射来访问,但反射访问性能不佳,在il2cpp下有可能因为代码剪裁而导致无法访问。

1.打标签

[LuaCallCSharp]
public class TestObj
{public void test(testStruct t){Debug.Log($"{t.a},{t.c}");}
}

2.静态列表

   [LuaCallCSharp]public static List<Type> luaCallCsharp = new List<Type>{typeof(TestObj),};

  上面的类加了此标签,就会生成适配代码

 2.ReflectionUse

一个C#类型类型加了这个配置,xLua会生成link.xml阻止il2cpp的代码剪裁。对于扩展方法,必须加上 LuaCallCSharp 或者 ReflectionUse 才可以被访问到。
建议所有要在Lua访问的类型,要么加LuaCallCSharp,要么加上ReflectionUse,这才能够保证在各平台都能正常运行。

如何配置?

[ReflectionUse]
public class TestReflectionUseObj
{}

link.xml保存着打上LuaCallCSharp和ReflectionUse标签的类

3.DoNotGen

指明一个类里头的部分函数、字段、属性不生成代码,通过反射访问。
只能标准 Dictionary<Type, List<string>> 的field或者property。key指明的是生效的类,value是一个列表,配置的是不生成代码的函数、字段、属性的名字。

如何配置?

[LuaCallCSharp]
public class TestDontGen
{public int a1;public int a2;
}

1.打标签(这种方式行不通)

在生成代码中,a2还是生成和a1一样的适配函数

2.静态列表

public static class ExportCfg
{[DoNotGen]public static Dictionary<Type,List<string>> dontGenDic = new Dictionary<Type, List<string>> {[typeof(TestDontGen)] = new List<string> { "a2" },};
}

4.CSharpCallLua

把一个lua函数适配到一个C# delegate(一类是C#侧各种回调:UI事件,delegate参数,比如List<T>:ForEach;另外一类场景是通过LuaTable的Get函数指明一个lua函数绑定到一个delegate),或者把一个lua table适配到一个C# interface时,使用该标签。

如何配置?

1.打标签:

   [CSharpCallLua]public interface IPlayerPosition{int x { get; set; }int y { get; set; }int z { get; set; }void add(int x, int y, int z);void sub(int x, int y, int z);}[CSharpCallLua]public delegate void AddMethod(LuaTable self, int x, int y, int z);[CSharpCallLua]public delegate Action addAc(LuaTable t, int x, int y, int z);[CSharpCallLua]public delegate void test1(int x);[CSharpCallLua]public delegate Action test2(int x);

2.静态列表:

[CSharpCallLua]
public static List<Type> CsharpCallLua = new List<Type> { typeof(IPlayerPosition),typeof(AddMethod),typeof(addAc),typeof(test1),typeof(test2),
};

 生成代码如下:接口生成独立文件,委托事件生成在DelegateGenBridge文件中

 

5.GCOptimize

xLua无参构造函数的复杂类型(struct)的默认传递方式是引用传递,当lua侧使用完毕释放该对象的引用时,会产生一次gc。
一个C#纯值类型(注:指的是一个只包含值类型的struct,可以嵌套其它只包含值类型的struct)或者C#枚举值加上这个标签之后,XLua会生成gc优化代码,在lua和c#间传递将不产生(C#)gc alloc,该类型的数组(一维?)访问也不产生gc。

如果没有配置该标签的值类型传递

public class Test{   [CSharpCallLua]public delegate Action testAction(TestGCOptimizeValue x);void Start(){luaenv = new LuaEnv();luaenv.AddLoader(customLoader);luaenv.DoString("require 'main'");ta = luaenv.Global.Get<testAction>("testAction");}private void Update(){if (Application.isPlaying){ta?.Invoke(ts);}}
}public struct TestGCOptimizeValue
{public int a;public int b;private float f;//[AdditionalProperties]public float F { get => f; set => f = value; }
}

生成的代码如下:

将struct类型装箱转换为object再进行处理。

如何配置?

[GCOptimize]
public struct TestGCOptimizeValue
{public int a;public int b;public float f;
}

 加上此标签后,就会在PackUnpack中生成适配代码

 

然后调用函数生成代码如下:

struct 满足条件如下:

1.struct允许嵌套其它struct,但它以及它嵌套的struct只能包含这几种基本类型:byte、sbyte、short、ushort、int、uint、long、ulong、float、double;例如UnityEngine定义的大多数值类型:Vector系列,Quaternion,Color。。。均满足条件,或者用户自定义的一些struct
2.该struct配置了GCOptimize属性(对于常用的UnityEngine的几个struct,Vector系列,Quaternion,Color。。。均已经配置了该属性),这个属性可以通过配置文件或者C# Attribute实现;
3.使用到该struct的地方,需要添加到生成代码列表;

6.AdditionalProperties

该标签GCOptimize的扩展配置,如果struct有的字段是私有的,需要通过属性来访问,这时就需要用到该配置(默认情况下GCOptimize只对public的field打解包)

如何配置?

如果不加此标签

[GCOptimize]
public struct TestGCOptimizeValue
{public int a;public int b;private float f;public float F { get => f; set => f = value; }
}

生成代码如下:无法生成访问f字段的代码

1.打标签方式

[GCOptimize]
public struct TestGCOptimizeValue
{public int a;public int b;private float f;[AdditionalProperties]public float F { get => f; set => f = value; }
}

 2.静态列表

  [AdditionalProperties]static Dictionary<Type, List<string>> AdditionalProperties = new Dictionary<Type, List<string>>{[typeof(TestGCOptimizeValue)] = new List<string> { "F" },};

生成代码如下:

7.BlackList

一个类型的一些成员不需要适配代码,可以通过加上该标签来实现。考虑到有可能需要把重载函数的其中一个重载列入黑名单,配置方式比较复杂,类型是 List<List<string>>,对于每个成员,在第一层List有一个条目,第二层List是个string的列表,第一个string是类型的全路径名第二个string是成员名如果成员是一个方法,还需要从第三个string开始,把其参数的类型全路径全列出来

如何配置?

[LuaCallCSharp]
public class TestBlackList
{public int a1;public int a2;public void Test(){}
}

若不加处理生成代码如下: 

1.打标签

C#代码:

[LuaCallCSharp]
public class TestBlackList
{public int a1;[BlackList]public int a2;[BlackList]public void Test(){}
}

生成代码如下: 

2.静态列表

列表中的第一个字符串是类型,第二个及以后是要屏蔽的字段

  [BlackList]public static List<List<string>> BlackList = new List<List<string>>(){new List<string>(){ "TestBlackList", "a2"},new List<string>(){ "TestBlackList", "Test"}};

生成代码:

注:
DoNotGen和ReflectionUse的区别是:1、ReflectionUse指明的是整个类;2、当第一次访问一个函数(字段、属性)时,ReflectionUse会把整个类都wrap,而DoNotGen只wrap该函数(字段、属性);
DoNotGen和BlackList的区别是:1、BlackList配了就不能用;2、BlackList能指明某重载函数,DoNotGen不能;

2.静态列表

有时我们无法直接给一个类型打标签,比如系统api,没源码的库,或者实例化的泛化类型,这时可以在一个静态类里声明一个静态字段,然后为这字段加上标签,并且这个字段需要放到一个静态类里,建议放到 Editor目录。

如何配置?

public static class ExportCfg
{[DoNotGen]public static Dictionary<Type, List<string>> dontGenDic = new Dictionary<Type, List<string>>{[typeof(TestDontGen)] = new List<string> { "a2" },};[BlackList]public static List<List<string>> BlackList = new List<List<string>>(){new List<string>(){ "TestBlackList", "a2"},new List<string>(){ "TestBlackList", "Test"}};[AdditionalProperties]public static Dictionary<Type, List<string>> AdditionalProperties = new Dictionary<Type, List<string>>{[typeof(TestGCOptimizeValue)] = new List<string> { "F" },};[LuaCallCSharp]public static List<Type> luaCallCsharp = new List<Type>{typeof(TestObj),};[CSharpCallLua]public static List<Type> CsharpCallLua = new List<Type> { typeof(IPlayerPosition),typeof(AddMethod),typeof(addAc),typeof(test1),typeof(test2),};
}

3.动态列表

声明一个静态属性,打上相应的标签即可。

见HotFix。

2.HotFix

1.标识要热更新的类型

1.打标签:(不建议,在高版本 Unity 不支持)

[Hotfix]
public class TestHotFix
{public void TestPrint(){Debug.Log("Before Hotfix");}
}

2.static列表:(高版本 Unity 需要把配置文件放 Editor 目录下)

[Hotfix]
public static List<Type> by_property
{get{return (from type in Assembly.Load("Assembly-CSharp").GetTypes()where type.Namespace == "XXXX"select type).ToList();}
}[Hotfix]
public static List<Type> by_field = new List<Type>()
{typeof(HotFixSubClass),typeof(GenericClass<>),
};

2.API

1.xlua.hotfix(class, [method_name], fix)

class :C#类,CS.Namespace.TypeName或者字符串方式"Namespace.TypeName"表示,字符串格式和C#的Type.GetType要求一致,如果是内嵌类型(Nested Type)是非Public类型,只能用字符串方式表示"Namespace.TypeName+NestedTypeName";
method_name : 方法名,可选;
fix : 如果传了method_name,fix将会是一个function,否则通过table提供一组函数。table的组织按key是method_name,value是function的方式。

2.base(csobj)

子类override函数通过base调用父类实现。
csobj : 对象
返回值 : 新对象,可以通过该对象base上的方法

3.util.hotfix_ex(class, method_name, fix)

可以在fix函数里头执行原来的函数缺点是fix的执行会略慢。
method_name : 方法名;
fix : 用来替换C#方法的lua function

Lua侧示例:

xlua.hotfix(CS.BaseTest, 'Foo', function(self, p)print('BaseTest', p)base(self):Foo(p)
end)

3.Hotfix Flag

使用方式:

[Hotfix(HotfixFlag.xxxx)]
public class TestHotFixClass
{}
1.Stateless、Stateful

使用xlua.util.state代替

2.ValueTypeBoxing

值类型的适配delegate会收敛到object(这是个啥?),好处是代码量更少,不好的是值类型会产生装箱,适用于对字段敏感的业务。

就是在生成delegate的时候,

如果是HotFix标签中包含ValueTypeBoxing,就会将方法和参数的类型转换为object类型

 

3.IgnoreProperty

不对属性注入及生成适配代码。

4.IgnoreNotPublic

不对非public的方法注入及生成适配代码。

5.Inline

不生成适配 delegate,直接在函数体注入处理代码。

6.IntKey

不生成静态字段,而是把所有注入点放到一个数组集中管理。

4.打补丁

Xlua可以用lua函数替换 C# 的构造函数,函数,属性,事件的替换。

1.函数

method_name 传函数名,支持重载,不同重载都是转发到同一个 lua 函数。

见Unity中 Xlua使用整理(一)_为xlua配置搜索路径-CSDN博客中的HotFix章节

2.构造函数

构造函数对应的 method_name 是 ".ctor"。和普通函数不一样的是,构造函数的热补丁是执行原有逻辑后调用lua。

3.属性

method_name 等于 get_AProp,setter 的 method_name 等于 set_AProp。

4.[]操作符

赋值对应 set_Item,取值对应 get_Item。第一个参数是 self,赋值后面跟 key,value,取值只有 key 参数,返回值是取出的值。

5.其他操作符

6.事件

7.析构函数

method_name 是 "Finalize",传一个 self 参数。析构函数的热补丁并不是替换,而是开头调用 lua 函数后继续原有逻辑。

8.泛型

其他规则一致,每个泛化类型实例化后都是一个独立的类型,只能针对实例化后的类型分别打补丁。

9.Unity协程

通过 util.cs_generator 可以用一个 function 模拟一个 IEnumerator,yield return使用在里头的 coroutine.yield模拟。

10.整个类

如果要替换整个类,使用一个 table,按 method_name = function 组织即可。

3.其他

 1.Unity协程的使用

 2.Async的使用

未完待续。。。

参考链接:

介绍 — XLua (tencent.github.io)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/66833.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Python Matplotlib教程-Matplotlib 多子图布局

Python Matplotlib 多子图布局 Matplotlib 是 Python 中最常用的数据可视化库&#xff0c;它提供了强大的功能来绘制不同类型的图表。在实际应用中&#xff0c;通常需要将多个图表绘制在同一个画布上&#xff0c;这就需要用到 多子图布局。本篇文章将详细介绍如何使用 Matplot…

全方位解读消息队列:原理、优势、实例与实践要点

全方位解读消息队列&#xff1a;原理、优势、实例与实践要点 一、消息队列基础认知 在数字化转型浪潮下&#xff0c;分布式系统架构愈发复杂&#xff0c;消息队列成为其中关键一环。不妨把消息队列想象成一个超级“信息驿站”&#xff0c;在古代&#xff0c;各地的信件、物资运…

【LLM】25.1.10 Arxiv LLM论文速递

25.1.9 12:00 - 25.1.10 12:00 共更新44 篇 —第1篇---- ===== AgroGPT: 高效的农业视觉-语言模型与专家调优 🔍 关键词: 农业, 视觉-语言模型, 大型多模态对话模型, 专家调优 PDF链接 摘要: 在推进大型多模态对话模型(LMMs)方面已经取得了显著进展,利用了在线可用的…

Scala语言的软件开发工具

Scala语言的软件开发工具 Scala是一种强类型、函数式编程语言&#xff0c;融合了面向对象和函数式编程的特性&#xff0c;近年来受到越来越多开发者的关注。在Scala生态系统中&#xff0c;有许多优秀的工具和框架可以帮助开发者提高生产力、简化开发流程。本文将探讨一些常用的…

[离线数仓] 总结三、Hive数仓DIM层开发

5.9 数仓开发之DIM层 DIM层设计要点: (1)DIM层的设计依据是维度建模理论,该层存储维度模型的维度表。 (2)DIM层的数据存储格式为orc列式存储+snappy压缩。 (3)DIM层表名的命名规范为dim_表名_全量表或者拉链表标识(full/zip)。 -- 数仓开发之DIM层 -- DIM层设计要点:…

VUE3 事件的处理

在 Vue.js 中&#xff0c;事件处理是通过 v-on 指令来实现的&#xff0c;允许我们在 DOM 元素上监听用户交互并执行相应的操作。通过事件绑定&#xff0c;Vue.js 可以响应用户的点击、输入、提交等行为。 1. 基本的事件绑定 v-on 指令用于监听事件&#xff0c;并在事件发生时…

Photon最新版本PUN 2.29 PREE,在无网的局域网下,无法连接自己搭建的本地服务器

1.图1为官方解答 2.就是加上这一段段代码&#xff1a;PhotonNetwork.NetworkingClient.SerializationProtocol SerializationProtocol.GpBinaryV16; 完美解决 unity 商店最新PUN 2 插件 不能连接 &#xff08;环境为&#xff1a;本地局域网 无外网情况 &#xff09; …

消息中间件类型介绍

消息中间件是一种在分布式系统中用于实现消息传递的软件架构模式。它能够在不同的系统或应用之间异步地传输数据&#xff0c;实现系统的解耦、提高系统的可扩展性和可靠性。以下是几种常见的消息中间件类型及其介绍&#xff1a; 1.RabbitMQ 特点&#xff1a; • 基于AMQP&#…

Github 2025-01-07Python开源项目日报 Top10

根据Github Trendings的统计,今日(2025-01-07统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Python项目10TypeScript项目1C++项目1OpenHands: 人工智能驱动的软件开发代理平台 创建周期:195 天开发语言:Python协议类型:MIT LicenseSta…

走进 JavaScript 世界:掌握核心技能

目录 一、引言 二、JavaScript 基础 &#xff08;一&#xff09;历史与特点 脚本语言 与 HTML 和 CSS 紧密结合 事件驱动和异步执行 &#xff08;二&#xff09;语法基础 变量与数据类型 运算符与表达式 控制结构 条件语句 循环语句 &#xff08;三&#xff09;函…

51单片机(二)中断系统与外部中断实验

中断即单片机因为某些原因E暂定现在的工作P0&#xff0c;转去做其他的工作P1&#xff0c;完了之后继续之前的事P0&#xff0c;其他工作P1就是中断程序&#xff0c;原因E就是中断事件&#xff0c;原因由外部发生&#xff0c;程序不能预测到的是硬中断&#xff0c;可以由程度触发…

python-42-使用selenium-wire爬取微信公众号下的所有文章列表

文章目录 1 seleniumwire1.1 selenium-wire简介1.2 获取请求和响应信息2 操作2.1 自动获取token和cookie和agent2.3 获取所有清单3 异常解决3.1 请求url失败的问题3.2 访问链接不安全的问题4 参考附录1 seleniumwire Selenium WebDriver本身并不直接提供获取HTTP请求头(header…

汽车信息安全 -- S32K1如何更新BOOT_MAC

目录 1.安全启动模式回顾 2.为什么要讨论BOOT_MAC 3.S32K1如何更新? 1.安全启动模式回顾 之前提到过,S32K1系列提供了Crypto Service Engine硬件加密模块(简称CSEc),大家可以通过该芯片系统寄存器SDID.FEATURES(System Device Identification Register)来判断自己的片子…

【Python】Python与C的区别

文章目录 语句结束符代码块表示变量声明函数定义注释格式Python的标识符数据输入input()函数数据输出print()函数 语句结束符 C 语言 C 语言中每条语句必须以分号;结束。例如&#xff0c;int a 10;、printf("Hello, World!");。分号是语句的一部分&#xff0c;用于…

理解Unity脚本编译过程:程序集

https://docs.unity3d.com/Manual/script-compilation.html 关于Unity C#脚本编译的细节&#xff0c;其中一个比较重要的知识点就是如何自定义Assembly。 预定义的assembly 默认情况下&#xff0c;Unity会按照这个规则进行编译。 PhaseAssembly nameScript files1Assembly-…

ubuntu22.04 gcc,g++从10.5切换到低版本9.5

一、安装gcc-9.5 mkdir gcc cd gcc sudo apt-get download $(apt-cache depends --recurse --no-recommends --no-suggests --no-conflicts --no-breaks --no-replaces --no-enhances --no-pre-depends gcc-9 | grep -v i386 | grep "^\w") sudo dpkg -i *.deb sudo…

统计有序矩阵中的负数

统计有序矩阵中的负数 描述 给你一个 m * n 的矩阵 grid&#xff0c;矩阵中的元素无论是按行还是按列&#xff0c;都以非递增顺序排列。 请你统计并返回 grid 中 负数 的数目 示例 1&#xff1a; 输入&#xff1a;grid [[4,3,2,-1],[3,2,1,-1],[1,1,-1,-2],[-1,-1,-2,-3]]…

Linux内核TTY子系统有什么(6)

接前一篇文章&#xff1a;Linux内核TTY子系统有什么&#xff08;5&#xff09; 本文内容参考&#xff1a; Linux TTY子系统框架-CSDN博客 一文彻底讲清Linux tty子系统架构及编程实例-CSDN博客 linux TTY子系统(3) - tty driver_sys tty device driver-CSDN博客 Linux TTY …

深入理解 Python 的多进程编程 (Multiprocessing)

在 Python 中&#xff0c;multiprocessing 模块提供了多进程支持&#xff0c;是处理并发任务的一个核心工具。与多线程不同&#xff0c;多进程可以突破 GIL&#xff08;Global Interpreter Lock&#xff0c;全局解释器锁&#xff09;的限制&#xff0c;充分利用多核 CPU 进行并…

《代码随想录》Day31打卡!

《代码随想录》贪心算法&#xff1a;合并区间 本题的完整题目如下所示&#xff1a; 本题的完整思路如下所示&#xff1a; 1.本题依然是先对数组的左边界进行排序。将数组的第一个元素赋值给current。 2.遍历数组&#xff0c;判断current中的右边界和当前元素的左边界是否有重叠…