Lua单例模式在重新require时

假设我有一个类的包装器:

local wrapper = {}
local some_library = require('some-library')

wrapper._library = some_library.new(...)

return wrapper

如果我在不同的文件中require我的包装器,那么每次都会创建一个新的some_library实例吗?例如:

// file1
local file1 = {}
local wrapper = require('wrapper')
--向file1添加方法
return file1

// file2
local file1 = require('file1')
local wrapper = require('wrapper')

然后我执行lua file2

在这种情况下,wrapper被包含两次;一次是在file1中,file2正在require的文件中,另一次是在file2自身中。现在会有两个some-library实例吗?如果我只想要一个实例,该怎么创建单例模式呢?

点赞
用户4984564
用户4984564

简短回答:

no

长回答:

当您 require 一个文件时,Lua 首先会检查模块是否已被加载。如果没有加载过,它会执行该文件并将其返回值保存在一个全局表中以进行缓存。这意味着 a)某个库只需要被 一次 需要, b)对于包装器也是如此,因此您只有 一个 包装器用于 一个 库;而不是针对一个库有几个包装器或有几个带有各自库的包装器。

但请记住:如果您的 模块 返回的不是库代码,而是一个接着创建库代码的函数,那么只缓存该函数,但每次运行该函数时它都会返回一个不同的库对象副本。例如:

-- lib.lua
return function()
  { foo = function(bar) print("fooing a bar...") end }
end

-- program.lua
factory_1 = require "lib"
factory_2 = require "lib"

print(factory_1 == factory_2) --> 打印 "true"

lib_1 = factory_1()
lib_2 = factory_1() -- 不是错别字,就是两次调用factory_1

print(lib_1 == lib_2) --> 打印 "false"

而且,如果您真的很无聊,可以创建一个库(作为表),当它被调用为函数时(使用 __call 元方法)返回一个就像第一个一样的库的新实例。

2018-05-23 09:41:50