如何阻止 Lua 对象构造函数在实例之间共享嵌套表?
2019-11-29 2:43:56
收藏:0
阅读:131
评论:3
有人可以修复我的有缺陷的 Lua 对象系统吗?
- 不同的实例有不同的数字字段
- 但是……当我向我的初始化字段添加表格时,这些表格在不同的实例之间共享(见下面的示例)。
我认为我需要对初始字段进行深拷贝,但我看不到在哪里。下面是我的代码。有什么建议吗?
Object = {}
function Object:new (o)
-- o = deep_copy(o) or {} -- <== 这行不起作用
-- self = deep copy(self) -- <== 这行不起作用
o = o or {}
setmetatable(o, self)
self.__index = self
self.__tostring = show
return o
end
Account = Object:new{balance = 0,all={}}
function Account:push(v)
self.all[#self.all+1] = v
end
function Account:deposit(v)
self.balance = self.balance + v end
function Account:withdraw (v)
if v > self.balance then
error"insufficient funds" end
self.balance = self.balance - v
end
function show(i, str,sep)
str,sep = "{",""
for k,v in pairs(i) do
if type(v) ~= "function" then
str = str..sep..tostring(k)..":"..tostring(v)
sep = ", "
end
end
return str .. "}"
end
为了说明问题,下面有两个实例“a”和“b”
当我更新a,b的数字字段时,不同的实例得到不同的值。
但当我在一个实例“a”中更新all表时,它会改变另一个实例“b”中的表。
a=Account:new()
b=Account:new()
a:deposit(100)
b:deposit(200)
b:push(10)
b:push(20)
a:push(300)
print("a all", show(a), show(a.all))
print("b all", show(b), show(b.all))
输出应该是:
a all {balance:100} {3:300}
b all {balance:200} {1:10, 2:20}
但实际输出是:
a all {balance:100} {1:10, 2:20, 3:300}
b all {balance:200} {1:10, 2:20, 3:300}
点赞
用户1847592
如Joseph所说,你需要为每个对象创建不同的 all。
你可以编写函数 Account:new() 或编写 Object:new() 函数,该函数接受初始化器(这样您就不需要为每个类实现不同的 :new()):
Object = {__init = function(o) end}
function Object:new(o, init)
-- 在创建实例时:
-- o: 实例本身
-- 在创建类时:
-- o: 表格包含共享字段和方法
-- init: 此类的实例初始化器
o = o or {}
self.__init(o)
setmetatable(o, self)
self.__index = self
if init then
function o.__init(o)
self.__init(o)
init(o)
end
end
return o
end
Account = Object:new({}, function(o) o.all={}; o.balance=0; end)
function Account:push(v)
self.all[#self.all+1] = v
end
function Account:deposit(v)
self.balance = self.balance + v
end
function Account:withdraw (v)
if v > self.balance then
error "insufficient funds" end
self.balance = self.balance - v
end
function show(i, str, sep)
str, sep = "{", ""
for k, v in pairs(i) do
if type(v) ~= "function" then
str = str .. sep .. tostring(k) .. ":" .. tostring(v)
sep = ", "
end
end
return str .. "}"
end
Object.__tostring = show
a = Account:new()
b = Account:new()
a:deposit(100)
b:deposit(200)
b:push(10)
b:push(20)
a:push(300)
print("a all", show(a), show(a.all))
print("b all", show(b), show(b.all))
2019-11-29 06:28:34
用户12456742
如果您想继承自 Object,最简单的方法是为 Account 创建一个新的构造函数。这应该为每个对象设置一个balance和all,而不是在类本身上设置。
Object = {}
Account = {}
function Object:new(o)
o = o or {}
setmetatable(o, self)
self.__index = self
self.__tostring = show
return o
end
function Account:new(o)
o = o or Object:new(o)
setmetatable(o, self)
self.__index = self
o.balance = 0
o.all = {}
return o
end
a all {all:table: 0x55ab93ec70d0, balance:100} {1:300}
b all {all:table: 0x55ab93ec6ed0, balance:200} {1:10, 2:20}
2019-11-29 14:31:17
评论区的留言会收到邮件通知哦~
推荐文章
- Lua 虚拟机加密load(string.dump(function)) 后执行失败问题如何解决
- 我想创建一个 Nginx 规则,禁止访问
- 如何将两个不同的lua文件合成一个 东西有点长 大佬请耐心看完 我是小白研究几天了都没搞定
- 如何在roblox studio中1:1导入真实世界的地形?
- 求解,lua_resume的第二次调用继续执行协程问题。
- 【上海普陀区】内向猫网络招募【Skynet游戏框架Lua后端程序员】
- SF爱好求教:如何用lua实现游戏内调用数据库函数实现账号密码注册?
- Lua实现网站后台开发
- LUA错误显式返回,社区常见的规约是怎么样的
- lua5.3下载库失败
- 请问如何实现文本框内容和某个网页搜索框内容连接,并把网页输出来的结果反馈到另外一个文本框上
- lua lanes多线程使用
- 一个kv数据库
- openresty 有没有比较轻量的 docker 镜像
- 想问一下,有大佬用过luacurl吗
- 在Lua执行过程中使用Load函数出现问题
- 为什么 neovim 里没有显示一些特殊字符?
- Lua比较两个表的值(不考虑键的顺序)
- 有个lua简单的项目,外包,有意者加微信 liuheng600456详谈,最好在成都
- 如何在 Visual Studio 2022 中运行 Lua 代码?

你可以使用
self.__index = self,并且你的原型有一个all属性,但你从未在你创建的对象中设置过。因此,访问它(如push操作所做的)将始终通过__index并最终命中原型。