3

有没有办法调用requireLua 文件,并让模块设置调用它的文件的环境?例如,如果我有一个定义函数RootSequence在表中定义的 DSL(域特定语言),我是否可以setfenv(1, dslEnv)在模块中有类似的东西允许我像全局变量一样访问这些函数?

我想到的目标是使用行为树 DSL,使我的定义文件看起来像这样(或尽可能接近):

require "behaviortrees"

return Root {
    Sequence {
        Leaf "leafname",
        Leaf "leafname"
    }
}

不必明确地将Root,Sequence和 和Leaf明确地纳入范围,也不必对 . 之类的名称进行限定behaviortrees.Sequence

简而言之,我试图使定义文件尽可能干净,没有任何多余的行使树定义混乱。

4

3 回答 3

3

我可以setfenv(1, dslEnv)在模块中有类似的东西允许我访问像全局变量这样的函数吗?

你当然可以。您只需要找出要使用的正确堆栈级别,而不是1在您的setfenv调用中使用。通常,您会使用带有debug.getinfo调用的循环沿堆栈向上走,直到在堆栈上找到require函数,然后再移动一些直到找到下一个主要块(以防万一有人调用require函数)。这是您必须使用的堆栈级别setfenv。但我可以建议一个...

不同的方法

require在 Lua 中是可插拔的。package.loaders您可以向数组添加一个函数(称为搜索器) ,并require在它尝试加载模块时调用它。假设您所有的 DSL 文件都有一个.bt后缀,而不是通常的.lua. 然后,您将使用普通 Lua 搜索器的重新实现,不同之处在于您将查找.bt文件而不是.lua文件,并且您将setfenv调用returnloadfile. 像这样的东西:

local function Root( x ) return x end
local function Sequence( x ) return x end
local function Leaf( x ) return x end


local delim = package.config:match( "^(.-)\n" ):gsub( "%%", "%%%%" )

local function searchpath( name, path )
  local pname = name:gsub( "%.", delim ):gsub( "%%", "%%%%" )
  local msg = {}
  for subpath in path:gmatch( "[^;]+" ) do
    local fpath = subpath:gsub( "%?", pname ):gsub("%.lua$", ".bt") -- replace suffix
    local f = io.open( fpath, "r" )
    if f then
      f:close()
      return fpath
    end
    msg[ #msg+1 ] = "\n\tno file '"..fpath.."'"
  end
  return nil, table.concat( msg )
end


local function bt_searcher( modname )
  assert( type( modname ) == "string" )
  local filename, msg = searchpath( modname, package.path )
  if not filename then
    return msg
  end
  local env = { -- create custom environment
    Root = Root,
    Sequence = Sequence,
    Leaf = Leaf,
  }
  local mod, msg = loadfile( filename )
  if not mod then
    error( "error loading module '"..modname.."' from file '"..filename..
           "':\n\t"..msg, 0 )
  end
  setfenv( mod, env ) -- set custom environment
  return mod, filename
end


table.insert( package.loaders, bt_searcher )

如果你把它放在一个模块中并且require它从你的主程序中一次,你可以然后require你的 DSL 文件与自定义环境从.bt文件中你也可以放置你的.lua文件的地方。而且您甚至不需要require("behaviortrees")DSL 文件中的 。例如:

文件xxx.bt

return Root {
  Sequence {
    Leaf "leafname",
    Leaf "leafname"
  }
}

文件main.lua

#!/usr/bin/lua5.1
require( "behaviortrees" ) -- loads the Lua module above and adds to package.loaders
print( require( "xxx" ) ) -- loads xxx.bt (but an xxx Lua module would still take precedence)
于 2017-08-01T06:55:19.403 回答
2

至少在 Lua 5.2 中,_ENV是一个确定环境表的局部变量。您可以更改任何函数的环境,基本上就是块。

_ENV = behaviortrees;

另一种方法是自动复制每个字段:

do
    _ENV = _ENV or _G;

    for k, v in next, behaviortrees do
        _ENV[k] = v;
    end
end

但是,手动本地化每个字段可能更有效behaviortrees

于 2017-07-31T23:43:26.990 回答
2

模块“behaviortrees.lua”

local behaviortrees = {
   -- insert your code for these functions
   Root     = function(...) ... end,
   Sequence = function(...) ... end,
   Leaf     = function(...) ... end,
}

-- Now set the environment of the caller.  Two ways are available:

-- If you want to make DSL environment isolated from Lua globals
-- (for example, "require" and "print" functions will not be available 
--  after executing require "behaviortrees")
setfenv(3, behaviortrees)
-- or 
-- If you want to preserve all globals for DSL
setfenv(3, setmetatable(behaviortrees, {__index = getfenv(3)}))

主要 Lua 程序:

require "behaviortrees"

return Root {
   Sequence {
      Leaf "leafname",
      Leaf "leafname"
   }
}
于 2017-08-01T06:50:08.840 回答