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

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

Python调用Orator ORM进行数据库操作

《Python调用OratorORM进行数据库操作》OratorORM是一个功能丰富且灵活的PythonORM库,旨在简化数据库操作,它支持多种数据库并提供了简洁且直观的API,下面我们就... 目录Orator ORM 主要特点安装使用示例总结Orator ORM 是一个功能丰富且灵活的 python O

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Nginx设置连接超时并进行测试的方法步骤

《Nginx设置连接超时并进行测试的方法步骤》在高并发场景下,如果客户端与服务器的连接长时间未响应,会占用大量的系统资源,影响其他正常请求的处理效率,为了解决这个问题,可以通过设置Nginx的连接... 目录设置连接超时目的操作步骤测试连接超时测试方法:总结:设置连接超时目的设置客户端与服务器之间的连接

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

Python使用国内镜像加速pip安装的方法讲解

《Python使用国内镜像加速pip安装的方法讲解》在Python开发中,pip是一个非常重要的工具,用于安装和管理Python的第三方库,然而,在国内使用pip安装依赖时,往往会因为网络问题而导致速... 目录一、pip 工具简介1. 什么是 pip?2. 什么是 -i 参数?二、国内镜像源的选择三、如何

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui

Linux使用nload监控网络流量的方法

《Linux使用nload监控网络流量的方法》Linux中的nload命令是一个用于实时监控网络流量的工具,它提供了传入和传出流量的可视化表示,帮助用户一目了然地了解网络活动,本文给大家介绍了Linu... 目录简介安装示例用法基础用法指定网络接口限制显示特定流量类型指定刷新率设置流量速率的显示单位监控多个