Lua中的元表,元方法

        元表其实也是一个表,只不过它以另一个表的查找表的形式存在,首先看下设置元表的两个方法

        setmetatable(table,metatable): 对指定table设置元表(metatable),如果元表(metatable)中存在__metatable键值,               setmetatable会失败 。返回值是一个设置了元表的对象(表)。

mytable = {}                          -- 普通表 
mymetatable = {}                      -- 元表
setmetatable(mytable,mymetatable)     -- 把 mymetatable 设为 mytable 的元表 

        getmetatable(table): 返回对象的元表(metatable)。

元表的定义:

在 Lua 的table 中我们可以访问对应的key来得到value值,但是却无法对两个 table 进行操作。

因此 Lua 提供了元表(Metatable),允许我们改变table的行为,每个行为关联了对应的元方法。

         例如,使用元表我们可以定义Lua如何计算两个table的相加操作a+b。

当Lua试图对两个表进行相加时,先检查两者之一是否有元表,之后检查是否有一个叫"__add"的字段,若找到,则调用对应的值。"__add"等即时字段,其对应的值(往往是一个函数或是table)就是"元方法"。

         再例如,当你通过键来访问 table 的时候,如果这个键没有值,那么Lua就会寻找该table的元表(假定有元表)中"__index "

        (1)第一种情况,如果__index包含一个表格,Lua会在表格中查找相应的键。

other = { foo = 3 }
table = setmetatable({}, { __index = other }) 
print(table.foo) --输出结果 3

因为table对应的foo键没有值,所以会查找__index对应的表,也就是other,other中存在
foo键,输出foo键对应的值,即3

         (2) 第二种情况,如果__index包含一个函数的话,Lua就会调用那个函数,table会作为参数传递给函数。

 __index 元方法查看表中元素是否存在,如果不存在,返回结果为 nil;如果存在则由 __index 返回结果。

mytable = setmetatable({key1 = "value1"}, {__index = function(mytable, key)
    if key == "key2" then
      return "metatablevalue"
    else
      return nil
    end
  end
})

print(mytable.key1,mytable.key2)

解析:

(1)mytable 表赋值为 {key1 = "value1"}。
(2)mytable 设置了元表,元方法为 __index。
(3)在mytable表中查找 key1,如果找到,返回该元素,找不到则继续。
(4)在mytable表中查找 key2,如果找到,返回该元素,找不到则继续。
(5)判断元表有没有__index方法,如果__index方法是一个函数,则调用该函数。
(6)元方法中查看是否传入 "key2" 键的参数(mytable.key2已设置),如果传
入 "key2" 参数返回 "metatablevalue",否则返回 mytable 对应的键值。

 我们可以将以上代码简单写成:
mytable = setmetatable({key1 = "value1"}, { __index = { key2 = "metatablevalue" } }) 
print(mytable.key1,mytable.key2)

 

以上所说的"__add","__index" 字段本身对应的值就是元方法,元方法可能被赋值为一个函数 ,也可能被赋值为另一个表。

 

一些常见元方法:

__add(a, b) --加法
__sub(a, b) --减法
__mul(a, b) --乘法
__div(a, b) --除法
__mod(a, b) --取模
__pow(a, b) --乘幂
__unm(a) --相反数
__concat(a, b) --连接
__len(a) --长度
__eq(a, b) --相等
__lt(a, b) --小于
__le(a, b) --小于等于
__index(a, b) --索引查询
__newindex(a, b, c) --索引更新(PS:不懂的话,后面会有讲)
__call(a, ...) --执行方法调用
__tostring(a) --字符串输出
__metatable --保护元表

【__newindex元方法】

__newindex元方法与__index类似,__newindex用于更新table中的数据,而__index用于查询table中的数据。当对一个table中不存在的索引赋值时,在Lua中是按照以下步骤进行的:

  1. Lua解释器先判断这个table是否有元表;
  2. 如果有了元表,就查找元表中是否有__newindex元方法;如果没有元表,就直接添加这个索引,然后对应的赋值;
  3. 如果有这个__newindex元方法,Lua解释器就执行它,而不是执行赋值;
  4. 如果这个__newindex对应的不是一个函数,而是一个table时,Lua解释器就在这个table中执行赋值,而不是对原来的table。
mymetatable={}
mytable=setmetatable({key1="value1"},{
	__newindex=mymetatable
	})
print(mytable.key1)   --value1
mytable.newkey="新值2"
print(mytable.newkey,mymetatable.newkey)--nil 新值2
mytable.key1="新值1"
print(mytable.key1,mymetatable.key1)--新值1  nil

【__add元方法】

--计算表的长度
function calTable(table)
	local count = 0
	for k,v in pairs(table) do
		count=count+1
	end
	return count
end
--__add:两表相加
mytable=setmetatable({1,2,3},{
	__add=function ( mytable1,newtable1)
		for i=1, calTable(newtable1) do
			table.insert(mytable1,newtable1[i])
		end
		return mytable1
	end
	})
secondTable={4,5,6}
mytable=mytable+secondTable
for k,v in pairs(mytable) do
	print(k,v)
end

 【__call元方法】

--__call:函数调用
mytable=setmetatable({10},{
	__call=function (mytable1,newtable1)
		local sum =0
		for i=1,calTable(mytable1) do
			sum=sum+mytable1[i]
		end
		for i=1,calTable(newtable1) do
			sum=sum+newtable1[i]
		end
			return sum
	end
	})
newtable={10,20,30}
print(mytable(newtable))  --输出70

【__tostring元方法】 

--__tostring:用于修改表的输出行为
mytable=setmetatable({10,20,30},{
	__tostring=function ( mytable )
		local sum = 0
		for k,v in pairs(mytable) do
			sum=sum+v
		end
		return "表中的所有元素和为:"..sum
	end
	})
print(mytable) --输出60

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值