首页 > 其他分享 >lua元表、元方法

lua元表、元方法

时间:2023-04-27 21:22:50浏览次数:50  
标签:__ 元表 mt -- tb lua 方法


lua元表、元方法

lua官方参考手册:https://www.runoob.com/manual/lua53doc/manual.html#2.4



一、总结:

☺ 1、普通的表,找不到了,或者无法进行运算的时候,考虑设置到它身上的元表的元方法

2、元表的本质:其实元表本质上就是普通的表,它只是在功能上和别人不一样!实际上,它还是一张普通的表

3、元表的作用:定义原始值在特定操作下的行为

4、元方法:其实就是元表里的一些函数



二、元表、元方法

1、元表的使用,举例子:

  • 当tb是一张表的时候,直接对它进行加法运算,会保错!
tb = {a = 1}
print(tb + 1)
  • 结果,会报错:attempt to perform arithmetic on a table value (global 'tb')
  • 解决:我们使用元表,定义一下,定义一个元方法 __add
tb = {a = 1}
-- 新建一个元表(元表其实就是普通的表,只是功能和普通表不一样)
mt = {
	-- 元表中的元方法__add
	__add = function(a, b)
		return a.a + b
	end
}
-- 给tb这张表绑定上元表,这样普通表的加法就被替换成元表的__add 方法
setmetatable(tb, mt)
print(tb + 1)
  • 结果:

    2


2、给普通的表设置上元表的函数 setmetatable(table, metatable)


3、元表决定了一个对象在数学运算、位运算、比较、连接、 取长度、调用、索引时的行为。

(1) 数学运算:加减乘除、取模、次方、取负、向下取整、按位与、按位或、按位异或、左移、右移

  • 接下来是元表可以控制的事件的详细列表。 每个操作都用对应的事件名来区分。 每个事件的键名用加有 '__' 前缀的字符串来表示; 例如 "add" 操作的键名为字符串 "_add"。

__add: + 操作。 如果任何不是数字的值做加法, Lua 就会尝试调用元方法。

比如下面的:tb 是表不是数字,不能直接做加法,lua就会去看看 tb 表是不是有设置上了元表mt,然后才会去看看元表mt中的元方法,发现是有加法__add 这个元方法的存在,然后就调用该加法元方法。

tb = {a = 1}
print(tb + 1)

4、重要且特殊的元方法 __index

(1) 作用:

table[key],当table不是表或是表table中不存在key这个键时,这个事件被触发。此时,会读出table相应的元方法。

(2) 举例子:

  • 当表tb中不存在索引是b的时候,tb[b] 最终会是nil
tb = {a = "hello"}
print(tb[b])
  • 解决:我们给tb设置上元表,然后元表定义一下,定义一个元方法 __index,这样当索引找不到的时候,就会去调用元方法
tb = {a = "hello"}
--定义一个元表
mt = {
	--定义一个元方法__index
	__index = function(table, key)
		return "hi,boy"	
	end

}
-- 给tb这张表绑定上元表
setmetatable(tb, mt)
print(tb['b'])
  • 结果:hi,boy

(3) __index 细节:这个事件的元方法,其实可以是一个函数也可以是一张表!

  • 比如上面的
--定义一个元表
mt = {
	--定义一个元方法__index
	__index = {
        a = 10,
        b = 5,
    }
}
  • 结果:print(tb['b']) 返回的是 5
  • 如果 print(tb['b']) 返回的是 hello,因为在tb表就可以找到,只有找不到,才会考虑设置在它身上的元表

5、元方法 __newindex

(1) 特点:赋值时触发

一旦有了"__newindex"元方法,Lua就不再做最初的赋值操作。(如果有必要,在元方法内部可以调用rawset方法来做赋值。)

  • 举例子:lua 原先的赋值方式如下:
tb = {a = "hello"}
tb['b'] = 10
print(tb['b'])
  • 结果是:10

  • 给tb表身上设置上元表,并且元表内有一个元方法 __newindex

tb = {a = "hello"}
--定义一个元表
mt = {
	--定义一个元方法__newindex
	__newindex = function(t, k, v) --因为————newindex 会覆盖给原来的表进行赋值的操作,即覆盖操作tb['b'] = 10
	end
}
setmetatable(tb, mt)
tb['b'] = 10
print(tb['b'])
  • 结果是:nil【因为在表身上设置上了元表,而元表存在了__newindex 方法,原先的赋值方式,就不生效了】
  • 解决:在元方法内部可以调用rawset方法来做赋值
tb = {a = "hello"}
--定义一个元表
mt = {
	--定义一个元方法__newindex
	__newindex = function(t, k, v)
		rawset(t,k,v) --发现,tb['b'] = 10 又可以正常赋值了
	end
}
setmetatable(tb, mt)
tb['b'] = 10
print(tb['b'])

(2) 为什么要使用rawset

  • 不使用rawset,可能会导致堆栈溢出


  • 使用rawset的原因:可以避免触发元方法__index



文章参考:

B站视频,作者-每日喝粥《【Lua】元表、元方法、面向对象》 https://www.bilibili.com/video/BV1f44y1a7Gk/




如果本文对你有帮助的话记得给一乐点个赞哦,感谢!

标签:__,元表,mt,--,tb,lua,方法
From: https://www.cnblogs.com/shan333/p/17360244.html

相关文章

  • Markov Chain Monte Carlo(MCMC) 方法
    MonteCarlo方法假设我们要求一个原函数并不明确的函数\(f(x)\)的在某个区间\([a,b]\)上的积分\(\theta=\int_{a}^bf(x)dx\)因为\(f(x)\)的原函数不知道,所以无法用牛顿-莱布尼茨公式计算。这里采用一种称为montecarlo的方法来模拟近似求解,它的思想如下,首先将待求的式子化......
  • Redis+lua 实现令牌桶限流算法
    使用lua:=redis.NewScript(script)args[0]=strconv.Itoa(fillInterval)args[1]=strconv.FormatInt(time.Now().Unix()*1000,10)res,err:=lua.Run(context.Background(),utils.Red,[]string{"RateLimit"},args[0],args[1]).Result()--定义返回值res[1]......
  • 自己动手实现Lua(一)
    二进制chunk:一段可以被lua解释器执行的代码就叫做chunk(写的代码)。预编译:为了获得更高的执行效率,Lua并不是直接解释执行chunk,而是先由编译器编译成内部结构(其中包含字节码等信息),然后再由虚拟机执行字节码。这种内部结构在Lua里就叫作预编译(Precompiled)chunk,由于采用了二进制格式,所......
  • Informix杀掉死锁的方法和查看死锁的sql语句
    Informix锁表产生的原因,要么是多个用户同时访问数据库导致该问题,要么是因为某个进程死了以后资源未释放导致的。如果是前一种情况,可以考虑将数据库表的锁级别改为行锁,来减少撞锁的机会;或在应用程序中,用setlockmodewait3这样的语句,在撞锁后等待若干秒重试。如果是后一种情况......
  • imagenomic portraiture安装方法, portraiture插件下载
    portraiture插件下载ImagenomicPortraiture是一款专业的人像磨皮插件,适用于AdobePhotoshop、AdobeLightroom和AppleAperture等软件。它提供了丰富的人像磨皮和美容修饰工具,可以帮助用户快速、高效地进行人像后期处理。imagenomicportraiture安装方法 下载好Portraiture......
  • java 后台给前台传值,html:select,html:text等加载页面显示默认值的方法
    后台写request.setAttribute("dateCreated","黑色头发");前台接收:html:text<html:textproperty="dateCreated"value="${dateCreated}"/>html:select<html:selectproperty="accountsUser"va......
  • Overload和Override的区别。Overloaded的方法是否可以改变返回值的类型?
    方法的重写Overriding和重载Overloading是Java多态性的不同表现。重写Overriding是父类与子类之间多态性的一种表现,重载如果在子类中定义某方法与其父类有相同的名称和参数,我们说该方法被重写如果在一个类中定义了多个同名的方法,它们或有不同的参数个数或有不同的参数类型,则称为方......
  • Python-字符串format方法指定参数
    一、字符串的format方法有几种指定参数的方式:(1)默认方式—传入的参数与{}一一对应(2)命名参数(3)未知参数{2}二、详细描述字符串的format方法如何格式化字符串:第一种方法:s1='Todayis{},thetemperatureis{}degrees.'print(s1.format('Saturday',24))第二种方法:s2='To......
  • 基于Linux系统的PXE搭建方法
    本文分享自天翼云开发者社区《基于Linux系统的PXE搭建方法》,作者:t***n 一、底层环境准备1、安装RedHat7.6系统2、关闭防火墙和Selinuxsystemctlstopfirewalldchkconfigfirewalldoffvim/etc/sysconfig/selinux    修改SELINUX=disabled3、配置本地yum源vim/e......
  • slmgr 命令提示“非核心”错误解决方法
    第一步:regeditHKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\WindowsNT\CurrentVersion\SoftwareProtectionPlatformSkipRearm的值修改为1,重启。第二步:slmgrslmgr-rearm以管理员身份运行上述命令后,重启。第三步:安装ipkslmgr/ipk[kms密钥] ......