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

相关文章

Nginx 访问控制的多种方法

《Nginx访问控制的多种方法》本文系统介绍了Nginx实现Web访问控制的多种方法,包括IP黑白名单、路径/方法/参数控制、HTTP基本认证、防盗链机制、客户端证书校验、限速限流、地理位置控制等基... 目录一、IP 白名单与黑名单1. 允许/拒绝指定IP2. 全局黑名单二、基于路径、方法、参数的访问控制

springboot3.x使用@NacosValue无法获取配置信息的解决过程

《springboot3.x使用@NacosValue无法获取配置信息的解决过程》在SpringBoot3.x中升级Nacos依赖后,使用@NacosValue无法动态获取配置,通过引入SpringC... 目录一、python问题描述二、解决方案总结一、问题描述springboot从2android.x

C#高效实现在Word文档中自动化创建图表的可视化方案

《C#高效实现在Word文档中自动化创建图表的可视化方案》本文将深入探讨如何利用C#,结合一款功能强大的第三方库,实现在Word文档中自动化创建图表,为你的数据呈现和报告生成提供一套实用且高效的解决方... 目录Word文档图表自动化:为什么选择C#?从零开始:C#实现Word文档图表的基本步骤深度优化:C

SpringBoot整合AOP及使用案例实战

《SpringBoot整合AOP及使用案例实战》本文详细介绍了SpringAOP中的切入点表达式,重点讲解了execution表达式的语法和用法,通过案例实战,展示了AOP的基本使用、结合自定义注解以... 目录一、 引入依赖二、切入点表达式详解三、案例实战1. AOP基本使用2. AOP结合自定义注解3.

Python中Request的安装以及简单的使用方法图文教程

《Python中Request的安装以及简单的使用方法图文教程》python里的request库经常被用于进行网络爬虫,想要学习网络爬虫的同学必须得安装request这个第三方库,:本文主要介绍P... 目录1.Requests 安装cmd 窗口安装为pycharm安装在pycharm设置中为项目安装req

nginx跨域访问配置的几种方法实现

《nginx跨域访问配置的几种方法实现》本文详细介绍了Nginx跨域配置方法,包括基本配置、只允许指定域名、携带Cookie的跨域、动态设置允许的Origin、支持不同路径的跨域控制、静态资源跨域以及... 目录一、基本跨域配置二、只允许指定域名跨域三、完整示例四、配置后重载 nginx五、注意事项六、支持

Qt实现对Word网页的读取功能

《Qt实现对Word网页的读取功能》文章介绍了几种在Qt中实现Word文档(.docx/.doc)读写功能的方法,包括基于QAxObject的COM接口调用、DOCX模板替换及跨平台解决方案,重点讨论... 目录1. 核心实现方式2. 基于QAxObject的COM接口调用(Windows专用)2.1 环境

MySQL查看表的历史SQL的几种实现方法

《MySQL查看表的历史SQL的几种实现方法》:本文主要介绍多种查看MySQL表历史SQL的方法,包括通用查询日志、慢查询日志、performance_schema、binlog、第三方工具等,并... 目录mysql 查看某张表的历史SQL1.查看MySQL通用查询日志(需提前开启)2.查看慢查询日志3.

MySQL底层文件的查看和修改方法

《MySQL底层文件的查看和修改方法》MySQL底层文件分为文本类(可安全查看/修改)和二进制类(禁止手动操作),以下按「查看方法、修改方法、风险管控三部分详细说明,所有操作均以Linux环境为例,需... 目录引言一、mysql 底层文件的查看方法1. 先定位核心文件路径(基础前提)2. 文本类文件(可直

Java实现字符串大小写转换的常用方法

《Java实现字符串大小写转换的常用方法》在Java中,字符串大小写转换是文本处理的核心操作之一,Java提供了多种灵活的方式来实现大小写转换,适用于不同场景和需求,本文将全面解析大小写转换的各种方法... 目录前言核心转换方法1.String类的基础方法2. 考虑区域设置的转换3. 字符级别的转换高级转换