Lua:将上下文传递到loadstring中?

我正在尝试传递上下文到我每次迭代 for 循环时计算的动态表达式中。我理解 loadstring 只在全局上下文中计算,这意味着本地变量不可访问。在我的情况下,我通过将本地变量转换为全局变量以便进行字符串计算来解决了这个限制。这是我的代码:

require 'cosmo'

model = { { player = "Cliff", age = 35, gender = "male" }, { player = "Ally", age = 36, gender = "female" }, { player = "Jasmine", age = 13, gender = "female" }, { 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>]]

...并生成不同的输出。目前我卡在了通过全局共享本地上下文的鹰派手段上。有没有人能够更好地了解我要做的事情如何实现?

原文链接 https://stackoverflow.com/questions/9268954

点赞
stackoverflow用户582
stackoverflow用户582

你可以使用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
        -- ...

这也可以防止条件值能够访问任何你不想让它访问的数据,或者更关键的是,修改任何你不希望修改的数据。然而,需要注意的是,你需要将任何顶级绑定放入到context表中,以便让condition能够访问(例如,如果你想让condition能够访问math库,则需要将其放入context中)。或者,如果你不希望让本地变量变成全局变量,而没有任何对condition访问的限制,你可以在context上使用metatable来将未知索引传递到_G

setmetatable(context, { __index = _G })
2012-02-13 22:32:04
stackoverflow用户7625
stackoverflow用户7625

值得注意的是,setfenv已从Lua 5.2中删除,loadstring已被弃用。5.2版本相对比较新,因此你暂时用不到,但是可以编写一个加载程序可以同时适用于两个版本:

local function load_code(code, environment)
    if setfenv and loadstring then
        local f = assert(loadstring(code))
        setfenv(f,environment)
        return f
    else
        return assert(load(code, nil,"t",environment))
    end
end

local context = {}
context.string = string
context.table = table
-- 等等,添加在应用程序中安全的库和函数。
-- 参见:http://lua-users.org/wiki/SandBoxes
local condition = load_code("return " .. args.condition, context)

版本5.2的load处理了旧的loadstring行为并设置了环境(即你的示例中的上下文)。版本5.2还改变了环境的概念,因此loadstring可能不是你最担心的问题。但是,这是值得考虑的,可能可以在未来节省一些工作。

2012-02-14 14:52:39