Lua:如何避免循环引用

问题

当我尝试进行循环 require 时,如何避免 Lua 5.1 给出以下错误?

$ lua main.lua
lua: ./bar.lua:1: loop or previous error loading module 'foo'
stack traceback:
    [C]: in function 'require'
    ./bar.lua:1: in main chunk
    [C]: in function 'require'
    ./foo.lua:1: in main chunk
    [C]: in function 'require'
    main.lua:1: in main chunk
    [C]: ?

文件结构

main.lua

require "foo"
require "bar"
print (Foo.getName())
print (Bar.getName())

foo.lua

require 'bar'
Foo = {}
Foo.name = 'foo'

function Foo:getName()
    return Foo.name .. Bar.name
end

bar.lua

require 'foo'
Bar = {}
Bar.name = 'bar'

function Bar:getName()
    return Bar.name .. Foo.name
end

期望输出

$ lua main.lua
foobar
barfoo
点赞
用户473961
用户473961

解决方案

main.lua

Foo = Foo or require "foo"
Bar = Bar or require "bar"
print (Foo.getName())
print (Bar.getName())

foo.lua

Foo = {}
Bar = Bar or require "bar"
Foo.name = 'foo'

function Foo:getName()
    return Foo.name .. Bar.name
end

return Foo

bar.lua

Bar = {}
Foo = Foo or require "foo"
Bar.name = 'bar'

function Bar:getName()
    return Bar.name .. Foo.name
end

return Bar

解释

由于您正在设置全局变量,因此在尝试另一个 require 之前,可以检查文件是否已被引用(即;全局变量是否已定义):

Bar = Bar or require "bar"

然后,您的 bar.lua 必须返回 Bar 的定义;

Bar = {}
-- ...
return Bar

这并不能完全解决问题,因为 bar.lua 期望定义 Foo。为了解决这个问题,您可以定义一个相同名称的虚拟变量:

Foo = {}
Bar = Bar or require "bar"

这仅在您将使用 Foo 推迟到调用函数时才可能。如果您想从 bar.lua 的范围内调用 Foo.name,则会遇到相同的循环依赖项问题。

2012-12-20 03:16:02
用户221509
用户221509

另一种解决这个问题的方法是改变代码的结构,并将“共同”的功能提取到一个第三方模块中,这样FooBar都需要使用它。

2012-12-20 10:18:31