【Unity/XLua】xlua自带教程示例分析(四)—— 使用C#接口和委托获取lua定义的表和对应特殊方法

第一步 创建Lua脚本

元表部分

lua 复制代码
lualocal calc_mt = {
    __index = {
        Add = function(self, a, b)
            return (a + b) * self.Mult
        end,
        
        get_Item = function(self, index)
            return self.list[index + 1]
        end,

        set_Item = function(self, index, value)
            self.list[index + 1] = value
            self:notify({name = index, value = value})
        end,
        
        add_PropertyChanged = function(self, delegate)
            if self.notifylist == nil then
                self.notifylist = {}
            end
            table.insert(self.notifylist, delegate)
            print('add', delegate)
        end,
                                
        remove_PropertyChanged = function(self, delegate)
            for i=1, #self.notifylist do
                if CS.System.Object.Equals(self.notifylist[i], delegate) then
                    table.remove(self.notifylist, i)
                    break
                end
            end
            print('remove', delegate)
        end,

        notify = function(self, evt)
            if self.notifylist ~= nil then
                for i=1, #self.notifylist do
                    self.notifylist[i](self, evt)
                end
            end	
        end,
    }
}
  • __index字段定义了元表的索引,即当通过实例访问一个字段或方法时,如果该实例本身没有这个字段或方法,Lua会查找元表中的__index来获取。
  • Add(self, a, b): 这是一个方法,接受两个参数 ab,返回它们相加后乘以实例的Mult字段的结果。
  • get_Item(self, index): 这个方法通过索引访问self.list表中的元素,索引从1开始。
  • set_Item(self, index, value): 这个方法用于设置self.list表中特定索引的值,并调用notify方法通知任何监听属性变化的回调函数。
  • add_PropertyChanged(self, delegate): 这个方法用于向notifylist中添加属性变化监听的回调函数。
  • remove_PropertyChanged(self, delegate): 这个方法用于从notifylist中移除指定的属性变化监听回调函数。
  • notify(self, evt): 这个方法用于触发notifylist中所有回调函数,并将事件对象evt传递给它们。

表部分

lua 复制代码
luaCalc = {
    New = function (mult, ...)
        print(...)
        return setmetatable({Mult = mult, list = {'aaaa','bbbb','cccc'}}, calc_mt)
    end
}
  • Calc表定义了一个方法New,这个方法接受一个参数mult和可变数量的其他参数)。

    方法返回了一个使用 calc_mt 作为元表的表。这个表包含两个字段:

    • Mult:传递给New方法的参数mult
    • list:一个包含三个字符串的数组。

第二步 创建C#脚本

首先声明与Lua的表结构对应的接口

cs 复制代码
public class PropertyChangedEventArgs : EventArgs
{
    public string name;
    public object value;
}

[CSharpCallLua]
public interface ICalc
{
    event EventHandler<PropertyChangedEventArgs> PropertyChanged;
    int Add(int a, int b);
    int Mult { get; set; }
    object this[int index] { get; set; }
}

[CSharpCallLua]
public delegate ICalc CalcNew(int mult, params string[] args);

首先是打头的PropertyChangedEventArgs,这是一个C#中和EventHandler配合使用的EventArgs类,用于表示参数

在lua中,其对应了set_item中的 self:notify({name = index, value = value}) 中输入的参数

其次是ICalc接口,该接口对应了luaCalc的返回值,其既声明了calc_mt元表中的字段,又声明了如Mult这个在luaCalc的New方法中才声明的字段。ICalc还重写了[]运算符

最后是委托方法,该方法对应了luaCalc的New函数,输入的参数一样,使用ICalc接口来接收对应值

接口和CalcNew方法都要打上CSharpCallLua标签,因为我们想要在C#中去访问和接收lua函数和返回值

第三步 lua执行过程操作

cs 复制代码
void Test(LuaEnv luaenv)
{
    luaenv.DoString(script);
    CalcNew calc_new = luaenv.Global.GetInPath<CalcNew>("Calc.New");
    ICalc calc = calc_new(10, "hi", "john"); //constructor
    Debug.Log("sum(*10) =" + calc.Add(1, 2));
    calc.Mult = 100;
    Debug.Log("sum(*100)=" + calc.Add(1, 2));

    Debug.Log("list[0]=" + calc[0]);
    Debug.Log("list[1]=" + calc[1]);

    calc.PropertyChanged += Notify;
    calc[1] = "dddd";
    Debug.Log("list[1]=" + calc[1]);

    calc.PropertyChanged -= Notify;

    calc[1] = "eeee";
    Debug.Log("list[1]=" + calc[1]);
}

void Notify(object sender, PropertyChangedEventArgs e)
{
    Debug.Log(string.Format("{0} has property changed {1}={2}", sender, e.name, e.value));
}

首先先使用DoString在环境中执行对应脚本,将其表都加载进虚拟机

接着创建一个声明了的委托对象calc_new接收Calc.New函数

再调用这个函数,并使用ICalc来接收构造好的表

接着便可以访问内部的对应函数和变量,如Add,Mult

值得注意的是在lua中声明的add_PropertyChanged,remove_PropertyChanged

在C#定义的接口中即会在事件PropertyChanged的 += 和 -= 时触发

其次是get_Item和set_Item,也会在C#中使用[]运算符重载时调用,算是一种命名规范

相关推荐
一个帅气昵称啊1 小时前
在.NET中使用RAG检索增强AI基于Qdrant的矢量化数据库
ai·性能优化·c#·.net·rag·qdrant
立刀人2 小时前
关于Unity 轴心点 Pivot、锚点 Anchor和控制轴
unity·游戏引擎
还是大剑师兰特3 小时前
C#面试题及详细答案120道(86-95)-- 进阶特性
c#·大剑师
我是唐青枫6 小时前
C#.NET ControllerBase 深入解析:Web API 控制器的核心基石
c#·.net
O败者食尘D6 小时前
【C#】使用Enigma将Winform或WPF打包成一个exe
c#
The Sheep 202310 小时前
C# 吃一堑,长一智
c#
q***829116 小时前
如何使用C#与SQL Server数据库进行交互
数据库·c#·交互
hixiong12318 小时前
C# OpenCVSharp实现Hand Pose Estimation Mediapipe
开发语言·opencv·ai·c#·手势识别
baivfhpwxf202319 小时前
SQL Server 服务端如何在其他电脑连接
c#
Dm_dotnet19 小时前
WPF/C#:使用Microsoft Agent Framework框架创建一个带有审批功能的终端Agent
c#