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

相关文章

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

Hadoop数据压缩使用介绍

一、压缩原则 (1)运算密集型的Job,少用压缩 (2)IO密集型的Job,多用压缩 二、压缩算法比较 三、压缩位置选择 四、压缩参数配置 1)为了支持多种压缩/解压缩算法,Hadoop引入了编码/解码器 2)要在Hadoop中启用压缩,可以配置如下参数

Makefile简明使用教程

文章目录 规则makefile文件的基本语法:加在命令前的特殊符号:.PHONY伪目标: Makefilev1 直观写法v2 加上中间过程v3 伪目标v4 变量 make 选项-f-n-C Make 是一种流行的构建工具,常用于将源代码转换成可执行文件或者其他形式的输出文件(如库文件、文档等)。Make 可以自动化地执行编译、链接等一系列操作。 规则 makefile文件

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

如何在页面调用utility bar并传递参数至lwc组件

1.在app的utility item中添加lwc组件: 2.调用utility bar api的方式有两种: 方法一,通过lwc调用: import {LightningElement,api ,wire } from 'lwc';import { publish, MessageContext } from 'lightning/messageService';import Ca

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time