我正在尝试将上下文传递给动态表达式,我评估for循环的每次迭代.我知道加载字符串仅在全局上下文中进行求值,这意味着局部变量是不可访问的.在我的情况下,我通过将本地转换为全局来进行字符串评估来解决此限制.这就是我所拥有的:
require 'cosmo' model = { { player = "Cliff",age = 35,gender = "male" },{ player = "Ally",age = 36,gender = "female" },{ player = "Jasmine",age = 13,{ player = "Lauren",age = 6.5,gender = "female" } } values = { eval = function(args) output = '' condition = assert(loadstring('return ' .. args.condition)) for _,it in ipairs(model) do each = it if condition() then output = output .. each.player .. ' age: ' .. each.age .. ' ' .. '\n' end end return output end } template = "$eval{ condition = 'each.age < 30' }" result = cosmo.fill(template,values) print (result)
我的最终目标(除了掌握Lua之外)就是构建一个像诱人引擎这样的XSLT,我可以做以下事情:
apply_templates{ match = each.age > 30}[[<parent-player>$each.player</parent-player>]] apply_templates{ match = each.age > 30}[[<child-player>$each.player</child-player>]]
……并产生不同的输出.目前,我一直坚持通过全球分享当地情境的强硬态度.这里有没有人能更好地了解我将如何做我正在尝试做的事情?
解决方法
您可以使用
setfenv()
更改函数的上下文.这使您基本上可以将加载的函数沙盒化到其自己的私有环境中.像下面这样的东西应该工作:
local context = {} local condition = assert(loadstring('return ' .. args.condition)) setfenv(condition,context) for _,it in ipairs(model) do context['each'] = it if condition() then -- ...
这还可以防止条件值访问您不想要的任何数据,或者更重要的是,修改您不希望它的任何数据.但是,请注意,您需要将任何顶级绑定公开到您希望条件能够访问的上下文表中(例如,如果您希望它能够访问数学包,那么您需要坚持使用进入上下文).或者,如果您对具有全局访问权限的条件没有任何问题,并且您只想处理不使本地变为全局的问题,则可以在上下文中使用Metatable使其将未知索引传递到_G:
setMetatable(context,{ __index = _G })