Lua:使用元表实现的一种面向对象方法调用

2024-03-25 16:48

本文主要是介绍Lua:使用元表实现的一种面向对象方法调用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Lua:使用元表实现的一种面向对象方法调用


一、Lua中的面向对象编程

Lua中,面向对象编程主要是通过table来实现的。

Lua中,定义对象及方法:

  • 冒号定义,冒号引用
local obj = {}function obj:setname(name)self.name = name
endfunction obj:getname()return self.name
endobj:setname("test1280")
print(obj:getname())

或者:

  • 点号定义,点号引用
local obj = {}function obj.setname(self, name)self.name = name
endfunction obj.getname(self)return self.name
endobj.setname(obj, "test1280")
print(obj.getname(obj))

或者:

  • 冒号定义,点号引用
local obj = {}function obj:setname(name)self.name = name
endfunction obj:getname()return self.name
endobj.setname(obj, "test1280")
print(obj.getname(obj))

或者:

  • 点号定义,冒号引用
local obj = {}function obj.setname(self, name)self.name = name
endfunction obj.getname(self)return self.name
endobj:setname("test1280")
print(obj:getname())

使用(冒号或点号)(定义或引用),区别在于是否将对象(table)作为第一个参数传入(self、this)。

可见,冒号定义方法或引用方法,是Lua为我们实现的一种语法糖。使得我们不必显式地传入对象本身。

如果既想要通过点号引用方法,又不希望显式地将对象本身作为第一个参数传入方法,如何实现?

一种可行的方法是,通过元表实现。


二、Lua中的元表

Lua中的每个值都有一套预定义的操作集合,例如数字相加减,字符串比较,字符串连接等等。

可以通过元表来修改一个值的行为,使其在面对一个非预定义的操作时执行一个指定的操作。

Lua中的每个值都有一个元表。

table和userdata可以有各自独立的元表,其他类型的值则共享其类型所属的单一元表。

Lua在创建新table时不会创建元表。

可以通过setmetatable来设置一个值的元表;
可以通过getmetatable来获取一个值的元表;(元表是一个table)

在Lua中只能设置table的元表,在C中可以设置任何值的元表。

当访问一个table中不存在的字段时,通常会返回nil值;特别的,当这个table的元表有__index元方法时,最终返回结果是__index的返回值。

例如:

local mt = {}
mt.__index = function (t, k)return rawget(t, "_" .. k)
endlocal obj = {}
setmetatable(obj, mt) obj["_name"] = "test1280"
print(obj["name"])print(obj["xxxx"])

结果:

[test1280@node1 20190808]$ lua me.lua
test1280
nil

三、设置元表__index元方法,满足【点号引用对象方法】的需求

-- fn function name
-- fv function valuelocal mt = {}
mt.__index = function (t, k)-- obj.fn->obj._fnlocal fv = rawget(t, "_" .. k)if type(fv) ~= "function" thenreturnend-- fv  upvalue-- ... 可变形参-- 注意返回的是一个函数return function (...)return fv(t, ...) -- 强制将t注入到第一个参数end
endlocal obj = {}
setmetatable(obj, mt)function obj._setname(this, name)this.name = name
endfunction obj._getname(this)return this.name
endobj.setname("test1280")
print(obj.getname())

1.设置obj对象的元表,包含__index元方法;

2.obj.setname的过程:

obj本身没有setname的字段,因此触发__index元方法;触发元方法时,obj对象以及"setname"方法名(字符串)传入__index元方法(参数);在__index中尝试查询obj是否存在"_setname"函数;如果存在名字叫做"_setname"的方法(函数),则创建一个新的匿名函数并将其返回;obj.setname(或者obj["setname"])获取到新创建的函数;这个新创建的匿名函数的形参是...,可以接受任何变参,因为我们无法提前知悉类的每个方法的参数形式(且参数列表也不可能统一类型、数量...);然后在新匿名函数中通过upvalue的方式引用真实的方法(函数,_setname),强制注入obj对象为第一个参数;同时将形参...作为参数原封不动地传入upvalue真实的函数;

好吧,有点绕…

以上是我结合元表(__index元方法)和upvalue等原生机制实现的一种通过点号调用对象方法的方法。

虽然调用形式统一,但同时也需要付出性能降低、开销增大的代价。

有得必有失,每次创建匿名函数可能会影响到性能噢,使用时需要注意。

参考:

1.《Lua程序设计 第二版》

这篇关于Lua:使用元表实现的一种面向对象方法调用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/845619

相关文章

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

Qt中QGroupBox控件的实现

《Qt中QGroupBox控件的实现》QGroupBox是Qt框架中一个非常有用的控件,它主要用于组织和管理一组相关的控件,本文主要介绍了Qt中QGroupBox控件的实现,具有一定的参考价值,感兴趣... 目录引言一、基本属性二、常用方法2.1 构造函数 2.2 设置标题2.3 设置复选框模式2.4 是否

Qt中QUndoView控件的具体使用

《Qt中QUndoView控件的具体使用》QUndoView是Qt框架中用于可视化显示QUndoStack内容的控件,本文主要介绍了Qt中QUndoView控件的具体使用,具有一定的参考价值,感兴趣的... 目录引言一、QUndoView 的用途二、工作原理三、 如何与 QUnDOStack 配合使用四、自

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

springboot整合阿里云百炼DeepSeek实现sse流式打印的操作方法

《springboot整合阿里云百炼DeepSeek实现sse流式打印的操作方法》:本文主要介绍springboot整合阿里云百炼DeepSeek实现sse流式打印,本文给大家介绍的非常详细,对大... 目录1.开通阿里云百炼,获取到key2.新建SpringBoot项目3.工具类4.启动类5.测试类6.测

Python列表去重的4种核心方法与实战指南详解

《Python列表去重的4种核心方法与实战指南详解》在Python开发中,处理列表数据时经常需要去除重复元素,本文将详细介绍4种最实用的列表去重方法,有需要的小伙伴可以根据自己的需要进行选择... 目录方法1:集合(set)去重法(最快速)方法2:顺序遍历法(保持顺序)方法3:副本删除法(原地修改)方法4:

pytorch自动求梯度autograd的实现

《pytorch自动求梯度autograd的实现》autograd是一个自动微分引擎,它可以自动计算张量的梯度,本文主要介绍了pytorch自动求梯度autograd的实现,具有一定的参考价值,感兴趣... autograd是pytorch构建神经网络的核心。在 PyTorch 中,结合以下代码例子,当你

Python中判断对象是否为空的方法

《Python中判断对象是否为空的方法》在Python开发中,判断对象是否为“空”是高频操作,但看似简单的需求却暗藏玄机,从None到空容器,从零值到自定义对象的“假值”状态,不同场景下的“空”需要精... 目录一、python中的“空”值体系二、精准判定方法对比三、常见误区解析四、进阶处理技巧五、性能优化

使用Python构建一个Hexo博客发布工具

《使用Python构建一个Hexo博客发布工具》虽然Hexo的命令行工具非常强大,但对于日常的博客撰写和发布过程,我总觉得缺少一个直观的图形界面来简化操作,下面我们就来看看如何使用Python构建一个... 目录引言Hexo博客系统简介设计需求技术选择代码实现主框架界面设计核心功能实现1. 发布文章2. 加

SpringBoot集成Milvus实现数据增删改查功能

《SpringBoot集成Milvus实现数据增删改查功能》milvus支持的语言比较多,支持python,Java,Go,node等开发语言,本文主要介绍如何使用Java语言,采用springboo... 目录1、Milvus基本概念2、添加maven依赖3、配置yml文件4、创建MilvusClient