Lua元表(Metatable)简易教程

文章目录

  • 0.友情链接
  • 1.引言
  • 2.创建一个元表
    • 2.1.__tostring方法
    • 2.2.__add和__mul方法
    • 2.3.__index方法
    • 2.4.__call方法
  • 3.完整代码

0.友情链接

  • GitHUb上下载Lua编译器
  • Lua菜鸟教程中的元表介绍(较全,但功能性受限)
  • 博客园内元表的介绍(较详细)
  • 简书内元表的介绍(较简洁)

1.引言

\qquadLua语言是用C写的,Lua的元表类似于Python的类,但书写难度远比Python的Class大,加上Lua没有免费的Debugger,这个问题就让人很头疼。在此写一个可以使用多个“方法”的元表(Metatable),代码较短,但容易理解,希望能帮到大家。
\qquad简单理解一个Lua语言的Metatable(元表),Metatable是table的一个拓展,setmetatable是创建一个Metatable的函数,它有两个参数——原table和Metatable的属性、方法列表。我们从这两方面出发创建一个Metatable。不熟悉Luatable操作的读者,建议查看以下的简短的教程链接:
Lua的table简介

2.创建一个元表

vector = {} -- 空属性
a = setmetatable({1,2,3},vector) -- create vector a
b = setmetatable({4,5,6},vector) -- create vector b

\qquad这样我们就创建一个空属性的列表,对他们的操作和table是一样的,现在我们要定义table的属性和方法了(即vector),创建了属性和方法的Metatable相当于继承了table的一个Lua类,我们先从简单的开始。

2.1.__tostring方法

\qquad__tostring方法是Metatable转换为字符串的方法,print一个Metatable时,就查看Metatable有无此方法,若没有,则会打印出一个table的编号;若有,则会按照此方法进行。
\qquad在此介绍一个简单的方法,先定义一个有函数指针的Lua函数,再将这个函数指针加入vector,即完成了Metatable对__tostring方法的继承。值得注意的是,当作为vector类的方法时,v_print函数的参数vector传递的参数就和python的self一样,传递的是元表本身。

v_print = function(vector)  -- print(v_print)return table.concat(vector,',')end
vector = {v_print}
a = setmetatable({1,2,3},vector) -- create vector a
b = setmetatable({4,5,6},vector) -- create vector b
c = {1,2,3}
print(a)
print(b)
print(c)

运行这个Lua程序,得到的结果如下:

>lua -e "io.stdout:setvbuf 'no'" "EX5.lua" 
1,2,3
4,5,6
table: 00A00920
>Exit code: 0

如果你成功了,那么恭喜你,你已经掌握了一半元表的知识。

2.2.__add和__mul方法

\qquad实际上,元表的操作方法有很多,具体就是用操作符来代替调用函数的方法,我们在此只介绍两个——加法和乘法。按照之前的套路,先定义加法和乘法的函数,与之不同的是,加法和乘法都是双目运算符,因此参数数目是2.

v_print = 	function(vec)  -- print(v_print)return table.concat(vec,',')end
v_add = function(vec,new_vec)  -- vector_a + vector_blocal result = setmetatable({},vector)for i = 1,#new_vec doresult[i] = vec[i]+new_vec[i]endreturn resultend
v_dot =	function(vec,new_vec) --vector_a·vector_b (inner product)local product = 0for i,v in ipairs(vec) doproduct = product + v*new_vec[i]endreturn productend
vector = {__tostring=v_print,__add=v_add,__mul=v_dot}
a = setmetatable({1,2,3},vector) -- create vector a
b = setmetatable({4,5,6},vector) -- create vector b
print(a+b)
print(a*b)

运行结果如下:

>lua -e "io.stdout:setvbuf 'no'" "Test1.lua" 
5,7,9
32
>Exit code: 0

\qquad这里定义的加法,返回的仍然是Metatable类,所以打印出的结果仍然不是table的编号
其余的操作符方法类似,在此不再赘述

模式描述
__adda+b
__suba-b
__mula*b
__diva/b
__moda%b
__unm-a
__concat..\text{..}..
_eqa==b
__lta<b
__lea<=b

2.3.__index方法

\qquad官方文档对于__index方法的解释是,若在元表中未找到对应的key(键值),则调用__index方法查找,文字叙述较为生疏,下面是一个例子:

v_print = 	function(vec)  -- print(v_print)return table.concat(vec,',')end
v_sum = function(vec)  -- sum of a veclocal sum = 0for i,v in ipairs(vec) dosum = sum+vendreturn sumend
v_index = 	function(vec,value)  -- value method of a vecif value == "sum" thenreturn v_sum(vec)endend
vector = {__index = v_index,__tostring = v_print}
a = setmetatable({1,2,3},vector) -- create vec a
print(a)
print(a["sum"])

输出结果如下

>lua -e "io.stdout:setvbuf 'no'" "Test2.lua" 
1,2,3
6
>Exit code: 0

\qquad可以发现,元表a中并没有"sum"这个键,所有程序调用__index方法,成功找到了键"sum"对应的value(这里是一个函数返回的值)。但是值得注意的是,这个方法并不支持多参数的传入,如果需要,请看下文。

2.4.__call方法

\qquad__call方法容许元表像函数用于使用,类似像a(para1,para2)的形式,和前面的方法一样,第一个参数默认传递的是a本身,但在调用的时候省略(与python的方法调用一致)。还是一样,先看一段简短的程序加深理解。

v_call = function(vec,new_vec,value)  -- vec(...)if value == nil thenfor i,v in ipairs(new_vec) dovec[i] = vendreturn vecendif value == "mul" thenreturn v_mul(vec,new_vec)endend
v_print = 	function(vec)  -- print(v_print)return table.concat(vec,',')end
v_mul =	function(vec,new_vec) -- vec_a.*vec_b (dot multiply)local result = setmetatable({},vector)for i,v in ipairs(vec) doresult[i] = v*new_vec[i]endreturn resultend
vector = {__tostring=v_print,__call=v_call}
a = setmetatable({1,2,3},vector) -- create vector a
b = setmetatable({4,5,6},vector) -- create vector b
print(a)
print(b)
print(a(b,"mul"))

输出结果:

>lua -e "io.stdout:setvbuf 'no'" "Test3.lua" 
1,2,3
4,5,6
4,10,18
>Exit code: 0

\qquad我们使用__call方法串入了两个参数,一个是Metatable b,还有一个是方法“mul”,虽然他是字符串,但在v_call中,它表示调用了v_mul方法。

3.完整代码

\qquad在此附上完整的代码及输出结果,方便读者对照理解。

v_add = function(vec,new_vec)  -- vec_a + vec_blocal result = setmetatable({},vec)for i = 1,#new_vec doresult[i] = vec[i]+new_vec[i]endreturn resultend
v_call = function(vec,new_vec,value)  -- vec(...)if value == nil thenfor i,v in ipairs(new_vec) dovec[i] = vendreturn vecendif value == "mul" thenreturn v_mul(vec,new_vec)endend
v_print = 	function(vec)  -- print(v_print)return table.concat(vec,',')end
v_mul =	function(vec,new_vec) -- vec_a.*vec_b (dot multiply)local result = setmetatable({},vector)for i,v in ipairs(vec) doresult[i] = v*new_vec[i]endreturn resultend
v_dot =	function(vec,new_vec) --vec_a·vec_b (inner product)local product = 0for i,v in ipairs(vec) doproduct = product + v*new_vec[i]endreturn productend
v_sum = function(vec)  -- sum of a veclocal sum = 0for i,v in ipairs(vec) dosum = sum+vendreturn sumend
v_index = 	function(vec,value)  -- value method of a vecif value == "sum" thenreturn v_sum(vec)endend
vector = {__add=v_add,__call = v_call,__tostring=v_print,__mul=v_dot,__index=v_index}
a = setmetatable({1,2,3},vector) -- create vec a
b = setmetatable({4,5,6},vector) -- create vec b
print(a)
print(b)
print(table.concat(a+b,","))
print(a*b)
print(a["sum"])
print(a(b,"mul"))

输出结果:

>lua -e "io.stdout:setvbuf 'no'" "EX5.lua" 
1,2,3
4,5,6
5,7,9
32
6
4,10,18
>Exit code: 0

希望本文对您有帮助,感谢您的阅读。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/544455.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

面试官:说一下 final 和 final 的 4 种用法?

作者 | 王磊来源 | Java中文社群&#xff08;ID&#xff1a;javacn666&#xff09;转载请联系授权&#xff08;微信ID&#xff1a;GG_Stone&#xff09;重要说明&#xff1a;本篇为博主《面试题精选-基础篇》系列中的一篇&#xff0c;查看系列面试文章请关注我。Gitee 开源地址…

ruby 将字符串转为数组_Ruby程序将数组打印为字符串

ruby 将字符串转为数组将数组打印为字符串 (Printing an array as string) Given an array and we have to print it as a string in Ruby. 给定一个数组&#xff0c;我们必须在Ruby中将其打印为字符串。 Ruby provides you various alternatives for the single problem. The…

超定方程组的最小二乘解

\qquad看了很多关于最小二乘解的博客&#xff0c;事实上都没有找到自己想要的证明过程&#xff0c;后来学了矩阵函数时才彻底搞明白了这件事情&#xff0c;所以和大家简单分享如下&#xff1a; \qquad已知矩阵Amn(m&#xff1e;n)A_{mn}(m&#xff1e;n)Amn​(m&#xff1e;n)是…

面试官:int和Integer有什么区别?为什么要有包装类?

作者 | 磊哥来源 | Java面试真题解析&#xff08;ID&#xff1a;aimianshi666&#xff09;转载请联系授权&#xff08;微信ID&#xff1a;GG_Stone&#xff09;重要说明&#xff1a;本篇为博主《面试题精选-基础篇》系列中的一篇&#xff0c;查看系列面试文章请关注我。Gitee 开…

shell从小做起:将100以内整除3的数列出来

#!/bin/bash for i in $(seq 1 100) do a$[ $i%3 ] #注&#xff1a; 在取余的时候 需要 运算 所以需要加运算符号 $[ ] if [ $a -eq 0 ]; thenecho "$i" fi done转载于:https://blog.51cto.com/286577399/1676501

c# 用空格分割字符串_C#| 左用空格填充字符串

c# 用空格分割字符串PadLeft() method is a library method of the String class. It is used to pad the string from the left side with spaces. PadLeft()方法是String类的库方法。 它用于从左侧用空格填充字符串。 Syntax: 句法&#xff1a; string string.PadLeft(int …

innodb是如何存数据的?yyds

前言如果你使用过mysql数据库&#xff0c;对它的存储引擎&#xff1a;innodb&#xff0c;一定不会感到陌生。众所周知&#xff0c;在mysql5以前&#xff0c;默认的存储引擎是&#xff1a;myslam。但mysql5之后&#xff0c;默认的存储引擎已经变成了&#xff1a;innodb&#xff…

【MATLAB】卡尔曼滤波器的原理及仿真(初学者专用)

文章目录0.引言1.场景预设2.卡尔曼滤波器3.仿真及效果0.引言 \qquad本文参考了Matlab对卡尔曼滤波器的官方教程及帮助文档&#xff08;Kalman Filter&#xff09;。官方教程的B站链接如下&#xff0c;在此对分享资源的Up主表示感谢。(如不能正常播放或需要看中文字幕&#xff0…

Go实现查找目录下(包括子目录)替换文件内容

为什么80%的码农都做不了架构师&#xff1f;>>> 【功能】 按指定的目录查找出文件&#xff0c;如果有子目录&#xff0c;子目录也将进行搜索&#xff0c;将其中的文件内容进行替换。 【缺陷】 1. 没有过滤出文本文件 2. 当文件过大时&#xff0c;效率不高 【代码】…

卡诺模板_无关条件的卡诺地图

卡诺模板Till now, the Boolean expressions which have been discussed by us were completely specified, i.e., for each combination of input variable we have specified a minterm by representing them as 1 in the K-Map. But, there may arise a case when for a giv…

面试官:final、finally、finalize 有什么区别?

作者 | 磊哥来源 | Java面试真题解析&#xff08;ID&#xff1a;aimianshi666&#xff09;转载请联系授权&#xff08;微信ID&#xff1a;GG_Stone&#xff09;重要说明&#xff1a;本篇为博主《面试题精选-基础篇》系列中的一篇&#xff0c;查看系列面试文章请关注我。Gitee 开…

【Matlab】扩展卡尔曼滤波器原理及仿真(初学者入门专用)

文章目录0.引言及友情链接1.场景预设2.扩展卡尔曼滤波器3.仿真及效果0.引言及友情链接 \qquad卡尔曼滤波器&#xff08;Kalman Filter, KF&#xff09;是传感器融合&#xff08;Sensor Fusion&#xff09;的基础&#xff0c;虽然知乎、CSDN、GitHub等平台已有大量的学习资料&am…

Windows 8.1 升级到专业版

本例将一台 Windows 8.1 平板升级到专业版。升级前&#xff1a;升级的原因&#xff0c;是因为用户发现这台平板不能启用远程桌面管理。查看计算机属性&#xff0c;显示如下&#xff1a;从上面的信息可以看出&#xff0c;目前这台平板安装的不是专业版。具体是什么版本呢&#x…

【MATLAB】求点到多边形的最短距离

文章目录0.引言1.原理2.代码及实用教程0.引言 \qquad点与多边形的关系无非三种——内部、上、外部。本文定义点在多边形内部距离为负&#xff0c;点在多边形边上距离为0&#xff0c;到多边形外部距离为正。 1.原理 计算点到多边形的距离分为3个步骤&#xff1a; 判断点与多边…

绝了,66道并发多线程面试题汇总

&#x1f446;&#x1f3fb;一个专注于 Java 面试的原创公众号。我花了点时间整理了一些多线程,并发相关的面试题&#xff0c;虽然不是很多&#xff0c;但是偶尔看看还是很有用的哦&#xff01;话不多说&#xff0c;直接开整&#xff01;01 什么是线程&#xff1f;线程是操作系…

ruby array_Array.select! Ruby中的示例方法

ruby arrayArray.select&#xff01; 方法 (Array.select! Method) In this article, we will study about Array.select! Method. You all must be thinking the method must be doing something related to the selection of objects from the Array instance. It is not as …

【Python】mmSegmentation语义分割框架教程(自定义数据集、训练设定、数据增强)

文章目录0.mmSegmentation介绍1.mmSegmentation基本框架1.1.mmSegmentation的model设置1.2.mmSegmentation的dataset设置1.2.1.Dataset Class文件配置1.2.2.Dataset Config文件配置1.2.3.Total Config文件配置2.运行代码 3.展示效果图和预测X.附录X.1.mmSegmentation框架解释 X…

基本shell编程【3】- 常用的工具awk\sed\sort\uniq\od

awk awk是个很好用的东西&#xff0c;大量使用在linux系统分析的结果展示处理上。并且可以使用管道&#xff0c; input | awk | output1.首先要知道形式awk command file 如 awk {print $0} a.txt b.txt &#xff08;后面可以跟一个或多个文件&#xff09;2.command学习。c…

面试官:重写 equals 时为什么一定要重写 hashCode?

作者 | 磊哥来源 | Java面试真题解析&#xff08;ID&#xff1a;aimianshi666&#xff09;转载请联系授权&#xff08;微信ID&#xff1a;GG_Stone&#xff09;重要说明&#xff1a;本篇为博主《面试题精选-基础篇》系列中的一篇&#xff0c;关注我&#xff0c;查看更多面试题。…

rotate array_Array.rotate! Ruby中的示例方法

rotate arrayArray.rotate&#xff01; 方法 (Array.rotate! Method) In this article, we will study about Array.rotate! method. You all must be thinking the method must be doing something which is related to rotating certain elements. It is not as simple as it…