运算符重载不起作用。

阅读 Lua 程序设计 时,我尝试了该书中给出的运算符重载示例

Set = {}

mt = {}
mt.__add = Set.union

-- 使用给定列表的值创建一个新集合
function Set.new (l)
    local set = {}
    setmetatable (set, mt)
    for _, v in ipairs (l) do
        set [v] = true
    end
    return set
end

function Set.union (a, b)
    local result = Set.new {}
    for k in pairs (a) do result [k] = true end
    for k in pairs (b) do result [k] = true end
    return result
end

function Set.intersection (a, b)
    local result = Set.new {}
    for k in pairs (a) do result [k] = b[k] end
    return result
end

function Set.tostring (set)
    local l = {}
    for e in pairs (set) do
        l[#l + 1] = e
    end
    return "{" .. table.concat (l, ", ") .. "}"
end

function Set.print (s)
    print (Set.tostring (s))
end

s1 = Set.new {10, 20, 30, 50}
s2 = Set.new {30, 1}

Set.print (s1)
Set.print (s2)

s3 = s1 + s2
Set.print (s3)

但是在最新的 lua for windows 中,我遇到了以下错误

lua: C:\meta.lua:47: attempt to perform arithmetic on global 's1' (a table value)
stack traceback:
    C:\meta.lua:47: in main chunk
    [C]: ?
{30, 10, 20, 50}
{1, 30}
点赞
用户107090
用户107090

你需要将mt定义为一个合适的元表:

mt = { __add = Set.union, __mul = Set.intersection, __tostring = Set.tostring }
2012-10-14 19:06:25
用户12048
用户12048

你把这个任务安排得太早了:

mt.__add = Set.union

因为 Set.union 还没有初始化。

把这个代码段移到 Set.union 的下方就可以运行了。

同样的道理,如果你赋值了 mt.__mul,那么它应该在 Set.intersection 下面。

2012-10-14 21:32:31