如何在Lua中获取已加载的脚本的确切路径?

我想要做的是在编写Lua代码时执行“热重载”。

我的问题在于我不知道如何将修改后的文件与“package.loaded”表中的条目配对。虽然我可以猜测,但我想知道完整路径是否存储在任何地方。

假设有一个名为“mymodule.lua”的文件,当我们运行以下代码时:

local x = require('mymodule')

首先,Lua在“package.loaded”中查看是否已经加载。如果没有,那么Lua就会使用“LUA_PATH”查找模块,我们可以在运行时像这样看到它:

> print(package.path)
/usr/share/lua/5.3/?.lua;/usr/share/lua/5.3/?/init.lua;/usr/lib64/lua/5.3/?.lua;/usr/lib64/lua/5.3/?/init.lua;./?.lua;./?/init.lua

在上述模块加载后,我们可以通过查找“package.loaded”表中的条目来查看其是否已加载,例如:

> for i,v in pairs(package.loaded) do
>> print(i, v)
>> end
debug   table: 0x55fa3bfe8b80
table   table: 0x55fa3bfe51e0
math    table: 0x55fa3bfe7820
mymodule    table: 0x55fa3bfe60f0
_G  table: 0x55fa3bfe2bc0
utf8    table: 0x55fa3bfe8700
bit32   table: 0x55fa3bfe9070
os  table: 0x55fa3bfe68f0
package table: 0x55fa3bfe4860
io  table: 0x55fa3bfe54e0
coroutine   table: 0x55fa3bfe4f90
string  table: 0x55fa3bfe7560

我的处理热重载的方法是将该表中的条目置空“package.loaded.mymodule = nil”,然后再次重新加载脚本,例如require('mymodule')。这似乎可以正常工作-已重新加载模块。

但是,还存在一些限制......例如,如果该模块实际上是设置为“myproject/mymodule/init.lua”,并且像这样加载require('mymodule'),那么“package.loaded”表中的条目将仅为“mymodule”,而不是在此情况下更有帮助的“mymodule/init.lua”。我不知道需要监视哪个文件,而不是“mymodule.lua”

在我的情况下,我在C中跟踪文件。我的问题在于我必须想办法猜测“package.loaded”表中的哪个条目与哪个文件匹配。

我想知道的是是否有一个表包含已加载的任何.lua文件的路径,以便我可以只监视该列表。

也许查找“package.loaded”并不是一个正确的方法?

点赞
用户1847592
用户1847592

可以修改两个搜索器来收集模块文件路径到表 package.loadedfrom

-- 这个代码块应该放在主程序的开头
do
   assert(_VERSION:match"^Lua 5%.[2-4]$")  -- 仅适用于 Lua 5.2+
   package.loadedfrom = {}

   local function save(modname, loader, path)
      if type(loader) == "function" then
         package.loadedfrom[modname] = path
      end
      return loader, path
   end

   local orig_searcher_2 = package.searchers[2]
   package.searchers[2] =
      function(modname)
         return save(modname, orig_searcher_2(modname))
      end

   local orig_searcher_3 = package.searchers[3]
   package.searchers[3] =
      function(modname)
         return save(modname, orig_searcher_3(modname))
      end
end

-- 现在引入所有的模块
require"mymodule"
require"mymodule2"

-- 现在获取模块文件路径列表
for modname, filepath in pairs(package.loadedfrom) do
   print(modname, filepath)
end
2020-02-18 16:10:18