元表是lua的核心语法之一。
作用: 类似“继承”,将两个table关联到一起,也可以理解为“附加”。
方法介绍: setmetatable(table,metatable)
: 对指定 table 设置元表(metatable),如果元表(metatable)中存在__metatable 键值,setmetatable 会失败。 getmetatable(table)
: 返回对象的元表(metatable)。
代码:
mytable = {} -- 普通表
mymetatable = {} -- 元表
setmetatable(mytable,mymetatable) -- 把 mymetatable 设为 mytable 的元表
local temp = getmetatable(mytable) -- 获取mytable的元表
print(temp)
print(mymetatable)
输出内存地址相同,说明getmetatable
成功获取了mytable的元表mymetatable。
只添加元表无法直接访问到元表的内容,此时如果直接访问元表中内容会返回nil
。此时就需要使用到__index元方法。
作用: 设置元表的__index
索引,让这个索引指向元表自身,这样才能直接访问到元表内容。 当你通过键来访问 table 的时候,如果这个键没有值,那么lua就会寻找该table的metatable(假定有)中的__index
键。如果__index
包含一个table,lua会在表格中查找相应的键。 注意:当table找不到索引时访问的是元表中__index
指向的table,而不是元表。
代码:
mytable = {}
mymetatable = {}
--两种写法
--1.
setmetatable(mytable,mymetatable)
mymetatable.__index = mymetatable
--2.
setmetatable(mytable,{__index = mymetatable})
如果__index
包含的是函数的话,lua就会调用那个函数,当前table和键会作为参数传递给函数。 __index
元方法查看表中元素是否存在,如果不存在,返回结果为 nil;如果存在则由__index
返回结果。
代码:
mytable = setmetatable({key1 = "value1"}, {
__index = function(mytable, key) --这里不要纠结参数,实际传进来的就是self和键值
if key == "key2" then
return "metatablevalue"
else
return nil
end
end
})
print(mytable.key1,mytable.key2)
输出:value1 metatablevalue 等同于
mytable = setmetatable({key1 = "value1"}, { __index = { key2 = "metatablevalue" } })
print(mytable.key1,mytable.key2)
__newindex 元方法
__newindex
元方法用来对表更新,__index
则用来对表访问 。
作用: 当你给表的一个缺少的索引赋值,解释器就会查找__newindex
元方法,如果存在则调用这个函数而不进行赋值操作。 代码:
mymetatable = {}
mytable = setmetatable({key1 = "value1"}, { __newindex = mymetatable })
print(mytable.key1)
mytable.newkey = "新值2"
print(mytable.newkey,mymetatable.newkey)
mytable.key1 = "新值1"
print(mytable.key1,mymetatable.key1)
输出: value1 nil 新值2 新值1 nil
如果对已存在的索引键(key1)赋值,则不调用元方法 __newindex
。 这里使用rawset函数给mytable赋值。rawset赋值可以让table元方法失效。rawset函数介绍 代码:
mytable = setmetatable({key1 = "value1"}, {
__newindex = function(mytable, key, value)
rawset(mytable, key, "\""..value.."\"")
end
})
mytable.key1 = "new value"
mytable.key2 = 4
print(mytable.key1,mytable.key2)
输出: new value “4”
利用这种机制我们可以实现只读的 table。 代码如下:
function readOnly(t)
local proxy = {} --定义一个空表,访问任何索引都是不存在的,所以会调用__index 和__newindex
local mt = {
__index = t, ---__index 可以是函数,也可以是table,是table的话,调用直接返回table的索引值
__newindex = function(t,k,v)
error("attempt to update a read-only table",2)
end
}
setmetatable(proxy,mt)
return proxy
end
days = readOnly{"Sunday","Monday","Tuesday","Wednessday","Thursday","Friday","Saturday"}
print(days[1])
days[2] = "hello" --这一行就非法访问了
输入结果: Sunday attempt to update a read-only table
总结: 1、如果 __newindex 是一个函数,则在给 table 中不存在的字段赋值时,会调用这个函数,并且赋值不成功。 2、如果 __newindex 是一个 table,则在给 table 中不存在的字段赋值时,会直接给 __newindex的table 赋值。
为表添加操作符lua可以自定义表的操作符,类似C#。
作用:自定义操作符逻辑。
代码: 下面示例自定义+操作符。
function table_maxn(t)
local mn = 0
for k, v in pairs(t) do
if mn
关注
打赏
最近更新
- 深拷贝和浅拷贝的区别(重点)
- 【Vue】走进Vue框架世界
- 【云服务器】项目部署—搭建网站—vue电商后台管理系统
- 【React介绍】 一文带你深入React
- 【React】React组件实例的三大属性之state,props,refs(你学废了吗)
- 【脚手架VueCLI】从零开始,创建一个VUE项目
- 【React】深入理解React组件生命周期----图文详解(含代码)
- 【React】DOM的Diffing算法是什么?以及DOM中key的作用----经典面试题
- 【React】1_使用React脚手架创建项目步骤--------详解(含项目结构说明)
- 【React】2_如何使用react脚手架写一个简单的页面?