middleclass:为属性添加 getter-setter 支持
2016-10-6 7:53:44
收藏:0
阅读:124
评论:1
我正在尝试为属性声明添加自动支持,以便为类生成自动生成的 getter 和 setter。我使用 middleclass 库作为类的基础。我定义了一个根类来处理属性创建。然而,在测试中,只有 root 类的直接子类正常工作。其他子类会在 middleclass 代码的深处抛出“堆栈溢出”错误( [string "local middleclass = {..."]:82: stack overflow)。
我的代码如下:
local CBaseObject=class('CObjectBase');
function CBaseObject:initialize()
self._init=true;
end;
function CBaseObject:finalize()
self._init=false;
end;
function CBaseObject:_getter_setter(v)
return v;
end;
function CBaseObject:_gen_prop_cache()
rawset(self,'_properties',rawget(self,'_properties') or {});
end;
function CBaseObject:__index(k)
print('GET',k);
self:_gen_prop_cache();
local prop=self._properties[k];
if prop~=nil
then
local getter=self[prop[2] or '_getter_setter'];
return getter(self,prop[1]);
else return nil;end;
end;
function CBaseObject:__newindex(k,v)
print('ME',self.class.name);
print('SET',k,v);
self:_gen_prop_cache();
local prop=self._properties[k];
if prop==nil and self._init or prop
then
if prop==nil then prop={};self._properties[k]=prop;end;
local vv=prop[1];
if type(v)=='table' and #v<4
then
for i=1,3 do prop[i]=v[i];end;
else
prop[1]=v;
end;
local setter=self[prop[3] or '_getter_setter'];
prop[1]=setter(self,prop[1],vv);
else
rawset(self,k,v);
end;
end;
测试类:
local cls=CBaseObject:subclass('test');
function cls:initialize()
self.class.super.initialize(self);
self.health={1000,'_gethealth','_sethealth'};
self.ammo=100;
self:finalize();
end;
function cls:_sethealth(value,old)
print('WRITE:',value,old);
if value<0 then return old;else return value;end;
end;
function cls:_gethealth(value)
print('READ:',value);
return value/1000;
end;
local cls2=cls:subclass('test2');
function cls2:initialize()
self.class.super.initialize(self);
self.ammo=200;
self:finalize();
end;
function cls2:_sethealth(value,old)
print('WRITE_OVERRIDEN:',value,old);
return value;
end;
local obj=cls2(); --将其更改为 cls() 以获得可运行的示例。
obj.health=100;
obj.health=-100;
print(obj.health,obj._properties.health[1]);
print(obj.ammo,obj._properties.ammo[1]);
我在 https://repl.it/languages/lua 上运行了我的代码。那么,我的做法是否正确?是否有可能以与所使用的库兼容的更简单的方式添加属性支持?或者我应该使用其他的,那么应该用什么呢?
编辑: 经过实验证明,构造函数 self.class.parent.<method>(<...>) 是错误的罪魁祸首。我将所有这样的出现都替换为实际的父类。这似乎是唯一的问题,之后代码开始无误运行。
点赞
评论区的留言会收到邮件通知哦~
推荐文章
- 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 代码?

Enrique García Cota(middleclass 的创建者)启发了我一个不错的 实现 getter/setter 的方法 在 middleclass 中。他建议创建和使用 混入。
在测试/使用此混入配方时,我进行了一些微小的更改。我目前正在使用的混入代码如下:
-- properties.lua local Properties = {} function Properties:__index(k) local getter = self.class.__instanceDict["get_" .. k] if getter ~= nil then return getter(self) end end function Properties:__newindex(k, v) local setter = self["set_" .. k] if setter ~= nil then setter(self, v) else rawset(self, k, v) end end return Properties您需要为您的属性创建
function get_\* 和function set_\*(或者根据需要修改上面的字符串模式)。示例:
local class = require('middleclass') local Properties = require('properties') local Rect = class('Rect'):include(Properties) function Rect:initialize(x, y, w, h) self.x = x self.y = y self.w = w self.h = h end function Rect:get_right() return self.x + self.w end function Rect:set_right(right) self.x = self.x + right - self.right end r = Rect:new(10,10, 100, 100) print(r.right) -- 110 print(r:get_right()) -- 110 r.right = 200 print(r.right) -- 200 print(r.x) -- 100这样,您可以在任何想要具有属性的类中使用此混入并简单地创建
get_\* 和set_\* 函数。然而,他还说:
性能提示:值得注意的是,使用像示例中的
__index函数应该会对代码的性能产生重大影响(如果与__index表进行比较)。个人而言,经过一段时间的使用 getters 和 setters(由于我的 Python 偏见),我决定明确地编写代码,不再依赖它们。当然,这取决于您的代码是否需要关注性能。