MVVM的核心就是数据驱动,数据驱动的核心就是数据绑定。
我一直在思考,如何使用lua做一个数据绑定的功能,仔细思考一下,数据绑定需要做到的功能很简单,就是当一个数据改变时,能主动回调一个或多个函数就好了。但是实现起来却不是那么简单。。
这个问题我苦想了几个月了,也做了几个版本出来,(代码不贴了),但是都不够完美,主要思路是读写数据使用get和set函数,既然是主动调用函数,实现回调其他函数自然就没问题了,再简化一点无非是对每个字段都可以自动生成get和set函数,但是很明显,这种方案完全自己使用还行,一旦结合其他库(哪怕是官方库),就做不到主动回调函数了。(例如官方库的sort函数,使用“x=123”的形式为x赋值的,并不是调用“setX(123)”呀)
所以还需要解决的除了主动回调函数外,还得能hook住每次“=”的调用。
然后我去思考元表的方案,始终没能想出一个好的方案。(智商感人)
而就在今天晚上,真的是灵光乍现啊,是哪位天使大姐被我感动了,来给我指点迷津吗。
必须写篇博客,就当是还愿了。
实现方法:
元表的__index和__newindex两个元方法,会lua的应该都熟悉了,就不啰嗦了。这两个元方法会在读写表中字段的时候触发,但是触发这两个元方法都需要一个条件:表中不存在某个字段。
接下来就简单了,只要始终保持表中不存在要绑定的字段就好了。
我TM真是天才。。
话不多说,上代码,这是我今晚随手写下的代码,真正使用的时候再稍微封装一下就可以了,几乎完美:
local function bindable(init)
local t = {}
local mt
mt = {
bind____ = {}, __index = function(table, key)
return mt[key]
end, __newindex = function(table, key, value)
local v_old = mt[key]
if v_old == value then
return
end
mt[key] = value
local slots = mt.bind____[key]
if slots then
for _, v in ipairs(slots) do
v(value, v_old)
end
end
end
}
setmetatable(t, mt)
for k, v in pairs(init) do
t[k] = v
end
return t
end local function bind(table, key, func)
local binds = table.bind____
binds[key] = binds[key] or {}
local bind = binds[key]
bind[#bind+] = func
return #bind
end local t = bindable({
x = ,
y =
}) local tag = bind(t, "x", function(val, old)
print("x changed:", "new:", val, "old:", old)
end) t.x =
t.x =
------------------------------------------------
2016.2.17 更新
我封装了一个完整版,放在了https://github.com/Anti-Magic/rdatabinding
代码只有100行,支持各种花式绑定,也支持取消绑定