我需要一些Lua metatables的帮助,特别是AutomagicTables(http://lua-users.org/wiki/AutomagicTables)。只分配给未定义的表的能力非常好,我想保留这个功能。我的版本已放在一个函数中:
require("dataentry") -- Contains my age function
function AutomagicTable()
-- Create a new data table
-- from https://lua-users.org/wiki/AutomagicTables
local auto, assign
function auto(tab, key)
return setmetatable({}, {
__index = auto,
__newindex = assign,
parent = tab,
key = key
})
end
local meta = {__index = auto}
function assign(tab, key, val)
if val ~= nil then
local oldmt = getmetatable(tab)
oldmt.parent[oldmt.key] = tab
setmetatable(tab, meta)
tab[key] = val
else
return nil
end
end
return setmetatable({}, meta)
end
我想要的是传递一个默认表,以便在未定义字段时使用 - 如PIL第13章(https://www.lua.org/pil/13.4.3.html)中所述。这将允许我的数据结构中的计算和查找字段。以下是我想要使用的语法:
t_defaults = {
Age = age(table["DOB"]),
Sex = "Female",
}
t = AutomagicTable(t_defaults)
t.ID12345.DOB = "7/2/1965"
t.ID12346.DOB = "1/2/1945"
print("ID12345",t.ID12345.Sex,t.ID12345.DOB,t.ID12345.Age)
print("ID12346",t.ID12346.Sex,t.ID12346.DOB,t.ID12346.Age)
注意,在此代码age()中对当前表(见下文)的DOB字段的引用失败,因为表[“DOB”]为nil。如果在没有默认值的情况下运行此代码,Automagic将返回缺失值的表。
我可以在PIL第13章中的示例后面指定默认值,但语法很乱,一旦应用我松开了AutomagicTable功能(因为我已经分配了不同的元表):
-- Make a metatables
t_defaults = {}
t_defaults.__index = function (table, key)
local def = {
Age = age(table["DOB"]),
Sex = "Female"
}
return def[key]
end
-- Set new metatable - but now we can't make anymore Automagic tables
setmetatable(t.ID12345, t_defaults)
setmetatable(t.ID12346, t_defaults)
-- This will work
print("ID12345",t.ID12345.Sex,t.ID12345.DOB,t.ID12345.Age)
print("ID12346",t.ID12346.Sex,t.ID12346.DOB,t.ID12346.Age)
-- This assignment fails
t.ID12347.DOB = "12/12/1945"
不幸的是,我并不完全了解AutomagicTables代码,并且很难在AutomagicTable代码中添加所需的功能。
感激不尽的任何帮助。
加文
答案 0 :(得分:0)
我认为你在这里不需要全面自动化 只应自动创建用户(即深度为1的对象) 因此,可以使用更简单的逻辑:
local function age(DOB_str)
local m, d, y = (DOB_str or ""):match"^(%d+)/(%d+)/(%d+)$"
if m then
local t = {month = tonumber(m), day = tonumber(d), year = tonumber(y)}
local now = os.date"*t"
local now_year = now.year
now = os.time{year = now_year, month = now.month, day = now.day}
local lower_bound = math.max(0, now_year - t.year - 1)
local completed_years = -1 + lower_bound
t.year = t.year + lower_bound
repeat
completed_years = completed_years + 1
t.year = t.year + 1
until os.difftime(now, os.time(t)) < 0
return completed_years
end
end
-- Class "User"
local user_default_fields = {Sex = "Female"}
local user_mt = {__index =
function (tab, key)
if key == "Age" then -- this field is calculatable
return age(tab.DOB)
else -- other fields are constants
return user_default_fields[key]
end
end
}
-- The table containing all users with auto-creation
local users = setmetatable({}, {__index =
function (tab, key)
local new_user = setmetatable({}, user_mt)
tab[key] = new_user
return new_user
end
})
-- usage
users.ID12345.DOB = "12/31/1965"
users.ID12346.DOB = "1/2/1945"
print("ID12345", users.ID12345.Sex, users.ID12345.DOB, users.ID12345.Age)
print("ID12346", users.ID12346.Sex, users.ID12346.DOB, users.ID12346.Age)