繁体   English   中英

Lua:将上下文传递给loadstring?

[英]Lua: pass context into loadstring?

我正在尝试将上下文传递给动态表达式,我评估for循环的每次迭代。 我知道加载字符串仅在全局上下文中进行求值,这意味着局部变量是不可访问的。 在我的情况下,我通过将本地转换为全局来进行字符串评估来解决此限制。 这就是我所拥有的:

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>]]

......并产生不同的产出。 目前,我一直坚持通过全球共享当地背景的强硬态度。 这里有没有人能更好地了解我将如何做我正在尝试做的事情?

值得注意的是, 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
-- etc. add libraries/functions that are safe for your application.
-- see: http://lua-users.org/wiki/SandBoxes
local condition = load_code("return " .. args.condition, context)

5.2版的load处理旧的loadstring行为并设置环境(在您的示例中为context)。 版本5.2也改变了环境的概念,因此loadstring可能是您最不担心的。 尽管如此,还是可以考虑为自己节省一些工作。

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

这还可以防止条件值访问您不想要的任何数据,或者更重要的是,修改您不希望它的任何数据。 但是,请注意,您需要将任何顶级绑定公开到您希望condition能够访问的context表中(例如,如果您希望它能够访问math包,那么您需要坚持使用进入context )。 或者,如果您对具有全局访问权限的condition没有任何问题,并且您只想处理不使本地变为全局,则可以在context使用metatable将其传递给_G

setmetatable(context, { __index = _G })

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM