当前位置: 首页 > news >正文

lua(xlua)基础知识点记录一

1. 关于 (…) 操作符

编译阶段优化:

Lua 编译器会对常量字符串进行优化处理,将连续的字符串拼接操作 (…)
合并为单个字符串。这种优化仅适用于编译期确定的常量字符串,不适用于运行时生成的动态字符串。 示例:

local str = "Hello" .. " " .. "World".."!!"
print(str) -- 编译时等价于:local str = "Hello World"
--这段代码运行后只会产生一个字符,(..)拼接过程中不会产生临时字符串

运行时拼接:

当拼接操作涉及变量或动态值时,优化不会生效,拼接将在运行时执行。对于频繁的字符串拼接操作,建议使用 table.concat
以避免创建大量临时字符串导致的性能问题。 示例:

local a = "Hello"
local b = "World"
local c = "!!"
local str = a .. " " .. b.." "..c -- 运行时执行拼接
print(str)

2. 关于 table.concat

table.concat 要求数组元素必须是字符串或数字类型。数字会自动转换为字符串,其他类型(如 boolean、table 或
nil)会引发错误。

3. 关于 xLua 调用 C# 重载方法

数字类型匹配:

Lua 仅使用 number 类型表示数字,而 C#包含多种数值类型(int、float、double、long)。当调用重载方法时,xLua 会按方法定义顺序依次匹配参数类型。 需要注意的是int,float,double,long都属于number 所以即使传递的是1.1这种浮点型,但是c#中仍然会调到第一个匹配以上四个参数的函数。
示例:比如如下代码.lua调用会走到第一个也就是int型的重载函数中

//Main.cs
public class Main{static int c1, c2, c3, c4, c5,c6,c7;public static void Execute(int a){c1++;}public static void Execute(float a){c2++;}
}
--text.lua
CS.Main.Execute(1.1)

nil 类型

Lua 传 nil 会被c#转换为 null,这种情况下,对于引用类型参数的传递null,仍可以正常调用,匹配规则仍然是按照重载定义的顺序。

lua传递参数个数多于#方法参数个数

传递参数多余c#方法的参数上限会报错。

// C# 重载方法
public class MethodOverloading : MonoBehaviour
{public void Method(float num1){Debug.Log("Method(float num1)");}public void Method(int num1){Debug.Log("Method(int num1)");}public void Method(string num1){Debug.Log("Method(string num1)");}
}
local classObject = CS.MethodOverloading()
classObject:Method(10) -- 输出结果取决于方法定义顺序

注:修改方法定义顺序将影响最终的调用结果。

下面是关于Xlua的luaCallc#调用反射源码(非warp方式调用):

//xlua匹配函数代码:
public int Call(RealStatePtr L)
{try{//若仅存在一个重载方法,且该方法无默认参数且无需强制类型检查,则直接调用该重载。if (overloads.Count == 1 && !overloads[0].HasDefalutValue && !forceCheck) return overloads[0].Call(L);for (int i = 0; i < overloads.Count; ++i) //顺序遍历所有重载{var overload = overloads[i];if (overload.Check(L))//检查参数是否匹配当前重载{return overload.Call(L);}}return LuaAPI.luaL_error(L, "invalid arguments to " + methodName);}catch (System.Reflection.TargetInvocationException e){return LuaAPI.luaL_error(L, "c# exception:" + e.InnerException.Message + ",stack:" + e.InnerException.StackTrace);}catch (System.Exception e){return LuaAPI.luaL_error(L, "c# exception:" + e.Message + ",stack:" + e.StackTrace);}}
//xlua检查函数参数是否匹配代码:
public bool Check(RealStatePtr L)
{int luaTop = LuaAPI.lua_gettop(L);      // 获取 Lua 栈顶索引(参数总数)int luaStackPos = luaStackPosStart;     // 从指定起始位置开始检查参数for (int i = 0; i < checkArray.Length; i++)  // 遍历预定义的检查函数数组{// 1. 跳过最后一个参数(若为可变参数)if ((i == (checkArray.Length - 1)) && (paramsType != null))break;// 2. 检查参数缺失(非可选参数)if (luaStackPos > luaTop && !isOptionalArray[i])return false;  // 参数不足 → 检查失败// 3. 检查参数类型不匹配else if (luaStackPos <= luaTop && !checkArray[i](L, luaStackPos))return false;  // 类型不匹配 → 检查失败// 4. 移动栈指针(仅当参数存在或非可选时)if (luaStackPos <= luaTop || !isOptionalArray[i])luaStackPos++;  // 指向下一个参数}// 5. 处理可变参数(params)情况return paramsType != null ? (luaStackPos < luaTop + 1 ? checkArray[checkArray.Length - 1](L, luaStackPos) : true) : luaStackPos == luaTop + 1;}

4. 关于table的rehash时机

我们都知道table是有数组和哈希两部分组成,但是实际上数组部分和哈希部分都是使用数组数据结构实现的,也就是意味着他们的,只不过table的数组部分的数组存储的是tValue,而哈希部分存储的是一个Node节点。并且table初始化时如果没有指定数组的长度,那么数值和哈希部分的长度都是0。那么后续增加数据的时候当如果储存不下的时候就会触发扩容-rehash。并且rehash都是按照2的指数增长。

local a = {}
for k = 1,5 dotable.insert(a,k,k)
end
-- 以上代码执行后会触发四次rehash,分别时添加1,2,3,5的时候

5.关于lua中面向对象

lua中继承类的实例之间共享基类的静态属性。并且修改会实时同步所有实例。比如下段代码中的A的M变量,所有继承A的类的实例之间该属性是共享的。

---@class A
---@field M _A
local A = class("A")
A.M = {A = 1,B = true}

lua中同一个类的所有实例的方法是共享的,在 Lua 的面向对象实现中,T1 和 T2 作为 B
类的实例,​共享类的方法,但各自拥有独立的成员变量。这种设计是 Lua 面向对象编程的核心机制,其原理基于 ​元表(Metatable)​​和 ​**__index 元方法**。比如:下列代码的e是结果是true

---@class B:A
local B = class("B",A)function B:Execute()self.super.Execute(self)
endlocal T1 = B.new()
local T2 = B.new()
local e = T1.Execute==T2.Execute
http://www.dtcms.com/a/283416.html

相关文章:

  • gem install报错解析
  • 小程序中状态管理Redux
  • ROCK Robotic R3 Pro -替代L2,适配多款无人机,支持机载、手持、车载以及船载
  • DrissionPage:一款让网页自动化更简单的 Python 库
  • 使用defineExpose暴露子组件的属性和方法、页面生命周期onLoad和onReady的使用
  • 【AI论文】可追溯证据增强的视觉基础推理:评估与方法论
  • OSS文件上传解析失败,错误:文件下载失败的排查与解决
  • 61.第二阶段x64游戏实战-抓取Lua分析本地和跨图寻路
  • Harbor 和 Helm
  • 陆面、生态、水文模拟与多源遥感数据同化的实践技术应用
  • ACL实验(思科设备)
  • 游戏开发中防止“范围蔓延”
  • Oracle 数据库常见等待事件参数详解
  • YOLO算法原理
  • 2025年中国品牌全球化发展分析:中国品牌在社交渠道、电商平台及官网流量方面显著增长
  • 测试开发工作日常用的提示词分享
  • 探秘京东外卖幕后:地图轨迹技术探寻
  • Java+Ollama 本地部署 DeepSeek-R1 对话机器人:从 0 到 1 实战指南
  • 动态规划的无后效性与马尔可夫性质相似关系的说明
  • [Java安全】JDK 动态代理
  • 3D TOF 安全防护传感器
  • 低精度定时器 (timer_list) 和 高精度定时器 (hrtimer)
  • 切比雪夫距离
  • Python高级编程技巧探讨:装饰器、Patch与语法糖详解
  • 二叉树思想草稿
  • 关于SaaS业务模式及其系统架构构建的详细解析
  • RICE-YOLO:基于改进YOLOv5的无人机稻穗检测新方法
  • 【C语言网络编程】HTTP 客户端请求(发送请求报文过程)
  • 在UE中如何操作视图的大小,方位,移动
  • 16路串口光纤通信FPGA项目实现指南 - 第二部分(上)