lua:关于本地范围的模块导入

use*_*203 5 embed lua scope module local

有两个脚本文件包含以下脚本

//parent.lua
function scope()
    local var = "abc"

    require "child"
end

//child.lua
print(var)
Run Code Online (Sandbox Code Playgroud)

这样,child.lua将打印一个nil值,因为parent.lua中的作用域不会将其本地功能公开给模块.我认为会这样,因为在此范围内和var声明之后声明了require指令.我的愿望是将孩子的所有行完全注入父母.只是导出子脚本以提高可读性.我如何通过本地范围?loadfile()不起作用,也没有dofile().函数环境fenv不包含本地值.debug.setlocal()似乎无法创建新变量(也需要子进程中的接收器).除了重新编译脚本之外的任何方法?

Pau*_*nko 2

只要稍加努力就可以。如果您的变量child是真正的上值,您可以将它们“链接”到scope函数中的值。如果它们是全局变量(这里似乎是这种情况),您可以使用将它们映射到环境setfenv局部变量将它们映射到环境并填充该环境。

以下将abc按照您的预期打印(您可以更改loadstringloadfile具有相同的效果):

function vars(f)
  local func = debug.getinfo(f, "f").func
  local i = 1
  local vars = {}
  while true do
    local name, value = debug.getlocal(f, i)
    if not name then break end
    if string.sub(name, 1, 1) ~= '(' then vars[name] = value end
    i = i + 1
  end
  i = 1
  while func do -- check for func as it may be nil for tail calls
    local name, value = debug.getupvalue(func, i)
    if not name then break end
    vars[name] = value
    i = i + 1
  end
  return vars
end

function parent()
  local var = "abc"

  local child = loadstring("print(var)")

  local env = vars(2) -- grab all local/upvalues for the current function
  -- use these values to populate new environment; map to _G for everything else
  setmetatable(env, {__index = _G})
  setfenv(child, env)

  child()
end

parent()
Run Code Online (Sandbox Code Playgroud)

这都是针对 Lua 5.1 的,但在 Lua 5.2 中也是可能的。