Lua:动态确定对象是“类”还是“实例”

我正在从另一种语言进行远程过程调用,在这种语言中,我不知道我要调用的Lua函数是“类”函数还是“成员”函数。这很重要,因为我需要知道是否将self作为参数传递。有没有一种方法确定我要调用的函数是成员函数还是类函数?

Lua 服务器

function obj.get_sum(a, b, c)
    return a + b + c
end
function obj:get_name()
    return self.name
end

Python 客户端

objClass.get_sum(1, 2, 3)
objInstance.get_name()

objInstance.get_name()的情况下,我需要将self作为参数传递给Lua函数。

点赞
用户5675002
用户5675002

在 Lua 中函数没有任何区别。

它可以存储在“类”中,也可以是全局函数,或将“类成员”函数复制到本地变量 - 没有办法确定函数是类的一部分还是自由函数,因为 Lua 中没有类。可以构建一个行为类似类系统的系统,但它不是 Java/C++中的类。你可以在任何其他不相关的表上调用该函数,并且它将正常运行,只要在该表中找到所有需要的数据。

可以通过使用 debug.getlocal()查看函数的第一个本地变量来猜测,因为参数算作本地变量。如果看到“self”作为名称 - 有一个可能性,该函数是使用冒号语法(function obj:method())声明的,但这只是明确的function obj.method(self)声明的语法糖,并不是函数-类关系的保证。即使第一个参数为“self”,它也可能是带有名为“self”或点语法和显式“self”参数的自由函数。如果使用点语法声明了“成员”函数,并在对象上显式命名了另一个名称,则在“成员”函数中可能看不到“self”参数,或者debug.getlocal()甚至可能无法找到参数名称,因为 Lua 源代码文件已预编译为剥离调试信息的字节码。

例如:

-- 与任何类无关的“自由”函数
local generic_func = function(table, x)
    return table.pos + x
end

-- 一些类的对象
local obj = { pos = 1}

-- 与“pos”元素无关的一些不相关表
local unrelated = { pos = "42" }

-- 使用冒号语法定义隐式“self”参数的“method”
function obj:func1(x)
    return self.pos + x
end

-- 使用点语法定义“method”,显式命名第一个参数
function obj.func2(object, x)
    -- 根本不存在“self”
    return object.pos + x
end

-- 将“自由函数”放入“类”中
obj.func3 = generic_func

-- 无论如何调用都产生相同的结果
print(obj:func1(2))
print(obj.func1(obj, 2))
print(obj:func2(2))
print(obj.func2(obj, 2))
print(obj:func3(2))
print(obj.func3(obj, 2))
print(generic_func(obj, 2))

-- 从“obj“的“method”函数在不相关的表上运行
print(obj.func1(unrelated, 2))

看这里:http://ideone.com/qRby6s

2017-07-04 18:11:44