将数字作为对象,或存储数字的属性。

在设计代数方程模拟系统时,我有一个困境:我们不能将属性与数字关联起来,如果我将数字转换为一个带有“值”字段的表格,我可以重载算术运算符,但不能重载逻辑运算符,因为只有当两个操作数具有相同的元表时,该运算符才起作用,而我的用户将经常将“x”与数字进行比较。

例如,这是一个最简方程求解器系统:

x = 0
y = 0
eq1 = {function() return 2*x + 3*y end, rhs = 1 }
eq2 = {function() return 3*x + 2*y end, rhs = 2 }
p = {{x,y},{eq1, eq2}}
solve(p)

“solve()将处理表格“p”以获取方程组和rhs的所有系数。但是,重要的是,用户可以将属性关联到“x”和“y”,例如下限、上限。我尝试使用表格,

x = {val=0, lb=0, ub=3}
y = {val=1,lb=3,ub=5}
....

并为“x”和“y”编写元方法,使算术操作将在x.val和y.val上操作。然而,在脚本环境中,我们还需要将“x”与数字进行比较,即“if x>0 then ...”。我困在这里。一个丑陋的解决方案是要求用户在建模方程和脚本中随处使用x.val,y.val。这里是否有人有类似的需要将属性关联到数字,并且数字仍然可以用于算术/逻辑操作?

点赞
用户182402
用户182402
这样做可能有效:

x = {val = 10}

mt = {} mt.__lt = function (op1, op2) if (type(op1) == 'table') then a = op1.val else a = op1 end if (type(op2) == 'table') then b = op2.val else b = op2 end

return a < b

end

setmetatable(x, mt)

print(x < 5) -- 输出 false print(x < 15) -- 输出 true print(x < x) -- 输出 false print(5 < x) -- 输出 true


当然,你可以为其他运算符(如 `__add`、`__mul`、`__eq` 等)编写类似的方法。

如果你不想使用 `type()`/反射,你可以使用更加糟糕的技巧,利用一元减号是一元的这一事实:

mt = {} mt.__unm = function (num) return -(num.val) end mt.__lt = function (a, b) return -(-a) < -(-b) end

```

2015-12-19 13:59:19
用户3979429
用户3979429

如果你有访问调试库(debug library)的权限,那么这很简单,你有吗?

debug.setmetatable(0, meta)

meta 将是所有数字的元表。这将解决你的逻辑重载问题。

然而,如果你想给数字分配属性,有一种方法可以做到这一点,我写了一个快速的例子,说明如何这样做:

local number_props = { {val="hi"}, {val="hi2"} }

debug.setmetatable(0,{__index=function(self,k)return number_props[self][k]end})

print((1).val, (2).val)
2015-12-19 16:54:11