判断一个值是否可以被调用

请注意,这个问题只涉及纯 Lua。我没有访问任何模块或 C 侧的权限。此外,我不能使用 IO、OS 或 debug 库。

我想要做的是一个接收以下参数的函数:

  • 一个表示秒数的数字
  • 一个可调用的值

通过“可调用的值”,我的意思是一个可以被调用的值。这可以是:

  • 一个函数
  • 一个带有元表的表,该元表允许通过 __call 元方法进行调用

下面是一个可调用表的示例:

local t = {}
setmetatable(t, {
  __call = function() print("Hi.") end
})
print(type(t)) --> table
t() --> Hi.

这是函数:

function delay(seconds, func)
  -- 第二个参数被称为“func”,但它可以是任何可调用的东西。
  coroutine.wrap(function()
    wait(seconds) -- 这个函数在其他地方定义。它等待所要求的秒数。
    func() -- 调用函数/表。
  end)()
end

但是我有一个问题。如果参数“func”不可调用,我希望该函数抛出一个错误。

我可以检查它是否是函数。但是如果它是一个带有元表的表,它允许调用会怎样呢? 如果表的元表没有受到“__metatable”字段的保护,那么,我可以检查元表以了解它是否可调用,但如果受到保护,我该怎么做呢?

请注意,我也考虑尝试使用 pcall 调用“func”参数,以检查它是否可调用,但为此,我需要提前调用它。

基本上,这是问题:我需要知道一个函数/表是否可调用,但不能尝试调用它。

原文链接 https://stackoverflow.com/questions/7550319

点赞
stackoverflow用户734069
stackoverflow用户734069

一般来说,如果元表不希望你能够获取它(通过将 __metatable 定义为特殊值),那么你将无法从 Lua 获取它。

然而,如果你想要作弊,你可以使用 debug.getmetatable,它将返回与该对象关联的元表。


你不必过早地使用 pcall 调用任何东西。观察:

pcall(function(...) return PossibleFunction(...) end, <insert arguments here>)
2011-09-26 03:41:13
stackoverflow用户1728830
stackoverflow用户1728830

这种尝试改进 Nicol 的答案仍然存在一个问题,它需要调用表格,但它告诉我们提供的表格是否可调用。即使表格是可调用的,在执行 __call() 元方法期间发生错误时,pcall() 也会返回 false

local result = {pcall(PossibleFunction, ...)}

if not result[1] then
    if result[2]:match"^attempt to call" then
        error("提供的值不可调用。")
    else
        -- 表格可调用但失败。只需传递错误。
        error(result[2])
    end
end

-- 对结果进行一些操作:
for i = 2, #result do
    print(result[i])
end

然而,以这种方式检查错误消息并不感觉很“干净”(如果使用的 Lua 解释器已经被修改为本地化错误消息怎么办?)。

2016-04-12 06:05:38
stackoverflow用户3117346
stackoverflow用户3117346
function iscallable(x)
    -- 判断传入的参数是否是一个函数
    if type(x) == 'function' then
        return true
    -- 判断传入的参数是否是一个表
    elseif type(x) == 'table' then
        -- 下面这一段判断是否可以执行 __call 来实现调用表本身
        -- 返回表的元表
        local mt = debug.getmetatable(x)
        return type(mt) == "table" and type(mt.__call) == "function"
    else
        return false
    end
end

-- 返回 iscallable 函数
return iscallable

使用示例:

> = iscallable(function() end)
true
> = iscallable({})
false
> = iscallable()
false
> = iscallable(nil)
false
> x = {}
> setmetatable(x, {__call=function() end})
> = iscallable(x)
true

如果你没有访问 debug 库,可能无法完全准确判断,因为你可以搞乱元表。你可以使用 pcall,但很难正确地分离错误。即使在这里搜索特定的错误字符串,那也可能是因为调用的函数内部有些东西是不可调用的,而不是这个特定的值不可调用,如果有意义的话。

2019-11-11 03:17:51