码迷,mamicode.com
首页 > 其他好文 > 详细

cocos2d-Lua02Lua面向对象

时间:2016-04-07 10:55:21      阅读:275      评论:0      收藏:0      [点我收藏+]

标签:

1 Lua中的面向对象

1.1 问题

lua不是一个面向对象的语言,但是他又有面向对象的思想。lua中的面向对象是伪面向对象,伪面向对象就要用到table实现。

由table模拟了一个面向对象的编程, table其实也可以看做是一个对象,和其他语言对象一样有个自身的标示(self),具有自己的生命周期。本案例演示lua中的面向对象。

1.2 方案

首先用table实现面向对象。

然后通过self得到当前lable对象。

接下来通过 : 符号实现声明函数时的隐试参数self。

最后通过require返回对象,调用该对象方法。

1.3 步骤

实现此案例需要按照如下步骤进行。

步骤一:用table实现面向对象

首先定义一个table类型的数据Account,然后声明label对象,通过对象调用方法,代码如下所示:

  1. Accout = {balance = 0}
  2. functionAccout.count( v )
  3.     print ("value is "..v)
  4. end
  5. --声明label对象(变量)accout
  6. accout = Accout
  7. --通过对象调用方法
  8. accout.count(1000)

编辑器中的效果如图-13所示:

技术分享

图-13

步骤二:通过self得到当前lable对象

通过对象调用方法,显示的把accout这个table 这个对象传入到函数 (显示的传入 self),还可通过冒号调用的方式隐试的传入self对象代码如下所示:

  1. Accout = {balance = 10}
  2. --显示传入self
  3. functionAccout.count( self, v )
  4.     self.balance = self.balance + v;
  5.     print ("value is "..self.balance)
  6. end
  7. --声明label对象(变量)accout
  8. accout = Accout
  9. --通过对象调用方法, 显示的把accout这个 table 这个对象传入到函数 (显示的传入 self)
  10. accout.count(accout, 1000)
  11. --还可通过 : 调用的方式隐试的传入 self 对象
  12. accout:count(2000)

编辑器中的效果如图-14所示:

技术分享

图14

步骤三:通过冒号实现声明函数的隐式参数

通过冒号符号隐式传入self,还可通过冒号调用的方式隐式的传入self对象,代码如下所示:

  1. Accout = { balance = 10}
  2. --通过 : 符号, 隐试传入 self
  3. functionAccout:count( v )
  4.     self.balance = self.balance + v;
  5.     print ("value is "..self.balance)
  6. end
  7. --声明label对象(变量)accout
  8. accout = Accout
  9. --还可通过 : 调用的方式隐试的传入 self 对象
  10. accout:count(2000)

编辑器中的效果如图-15所示:

技术分享

图-15

步骤四:通过require 返回对象

通过require 返回对象,调用该对象方法,代码如下所示:

  1. class = require("myClass")
  2. --函数写在 table的括号外
  3. class:showName()
  4. --函数写在 table的括号内
  5. class.show(class)

编辑器中的效果如图-16所示:

技术分享

图-16


-------------》require(“XXX”)  --  只执行一次,避免了重复执行


1.4 完整代码

本案例中,完整代码如下所示:

  1. --[[lua不是一个面向对象的语言,但是他又有面向对象的思想
  2. lua中的面向对象是伪面向对象,伪面向对象就要用到table实现,
  3. 是table模拟了一个面向对象的编程, table其实也可以看做是一个对象,
  4. 和其他语言对象一样有个自身的标示(self),具有自己的生命周期。]]
  5. --1.用table实现面向对象
  6. Accout = {balance = 0}
  7. functionAccout.count( v )
  8.     print ("value is "..v)
  9. end
  10. --声明label对象(变量) accout
  11. accout = Accout
  12. --通过对象调用方法
  13. accout.count(1000)
  14. --2.通过 self 得到当前 lable 对象
  15. Accout = {balance = 10}
  16. --显示 传入 self
  17. functionAccout.count( self, v )
  18.     self.balance = self.balance + v;
  19.     print ("value is "..self.balance)
  20. end
  21. --声明label对象(变量) accout
  22. accout = Accout
  23. --通过对象调用方法, 显示的把 accout 这个 table 这个对象传入到函数 (显示的传入 self)
  24. accout.count(accout, 1000)
  25. --还可通过 : 调用的方式 隐试的传入 self 对象
  26. accout:count(2000)
  27. --3.通过 : 符号,实现声明函数时的 隐试参数 self
  28. Accout = { balance = 10}
  29. --通过 : 符号, 隐试 传入 self
  30. functionAccout:count( v )
  31.     self.balance = self.balance + v;
  32.     print ("value is "..self.balance)
  33. end
  34. --声明label对象(变量) accout
  35. accout = Accout
  36. --还可通过 : 调用的方式 隐试的传入 self 对象
  37. accout:count(2000)
  38. --4.通过require 返回对象, 调用该对象方法
  39. class = require("myClass")
  40. --函数写在 table的括号 外
  41. class:showName()
  42. --函数写在 table的括号 内
  43. class.show(class)
2、元方法与元表
元方法:就是两个 类型 变量之间进行操作的 函数
例如,1.数字的相加,它可能仅仅是一个函数:1 + 1--->add(1,1)--->add函数就是用来计算两个数字间相加的结果de
2.table类型相加?--->Lua中不存在两个{} + {}的元方法!
元表(metatable):就是用来存放 元方法 的 table,是table预定义的一系列操作。把两个table相加,那么Lua就会先去检查两个table是否有metatable,然后再检 查metatable中是否有__add方法。如果有按照__add方法中的操作来执行,否则,报错。
例如,如果非要将两个table类型相加的话,直接运行必报错。但Lua允许修改元表,如下所示:

一个元表,其实就是一个table值,所以,我们只需要新建一个table,添加元方法即可。

比如加法运算的元方法就是:__add,这是Lua规定的。

只要某个值的元表里含有__add这个元方法,那就可以使用+号进行运算。

 

如下代码:

  1.     -- 创建一个元表
  2.     local mt = {};
  3.     mt.__add = function(t1, t2)
  4.         print("两个table相加的结果就是...Good Good Study,Day Day Up!");
  5.     end
  6.     local t1 = {};
  7.     local t2 = {};
  8.    
  9.     -- 给两个table设置新的元表
  10.     setmetatable(t1, mt);
  11.     setmetatable(t2, mt);
  12.    
  13.     -- 进行加法操作
  14.     local result = t1 + t2;

首先创建了一个table变量mt,给这个table新增一个元素__add,这个table就拥有了作为元表的资格了。

然后创建两个新的table变量,使用setmetatable函数给table设置新的元表,此时,两个table变量就以mt作为元表了。

最后,对t1和t2进行加法操作,这时就会从元表中查找__add元方法,如果找到的话,就调用这个元方法对两个变量进行加法操作。


Lua中的每一个值都有或者可以有一个元表,但table和userdata可以拥有独立的元表,其他类型的值就只能共享其类型所属的元素。比如字符串使用的是
string的元表。注意:Lua在新建table时,不会创建metatable,需要使用setmetatable来设置元表。setmetatable的参数可以使任意table,包括要赋值的table本身。

------》查看一个变量是否有元表:
print(getmetatable(a)) -- nil 表示没有元表


Lua中的重要的元方法:__eq等于、__lt小于、__le小于等于、__add:加法、__sub:减法、__mul:乘法、__div:除法、__unm:相反数、__mod:取模、__pow:乘幂、__index function(table,key)、__newindex function(table,key,value)、__tostring 被print()调用、__metatable设置后可隐藏mt

只要在自定义元表的时候,给这些元方法名赋予新的函数就可以实现自定义操作了。

Lua访问table元素 首先,通过__index元方法来查找是否有这个函数(如果没有,返回nil);__index的值可以直接是一个table,也可以是函数(若是table,则以该table作为索引进行查询;若是函数,则将table和缺少的域作为参数,调用这个函数)


 例子1、

我们新建一个自定义的元表(也就是一个table变量),用来定义一些操作:

  1.     -- 创建一个元表
  2.     local mt = {};
  3.     mt.__add = function(s1, s2)
  4.         local result = "";
  5.         if s1.sex == "boy" and s2.sex == "girl" then
  6.             result = "青梅竹马";
  7.         elseif s1.sex == "girl" and s2.sex == "girl" then
  8.             result = "空对空";
  9.         else
  10.             result = "硬碰硬"
  11.         end
  12.        
  13.         return result;
  14.     end

 

其实这和上一节的例子基本一样,只是多说一次而已,使用方式如下:

  1.     -- 创建两个table,可以想象成是两个类的对象
  2.     local s1 = {
  3.         name = "Hello",
  4.         sex = "boy",
  5.     };  
  6.     local s2 = {
  7.         name = "Good",
  8.         sex = "girl",
  9.     };
  10.    
  11.     -- 给两个table设置新的元表
  12.     setmetatable(s1, mt);
  13.     setmetatable(s2, mt);
  14.    
  15.     -- 进行加法操作
  16.     local result = s1 + s2;
  17.    
  18.     print(result);
例子2、
Window = {}
Window.mt = {}
Window.prototype = {x = 0,y = 0,width = 100,height = 100}
Window.mt.__index = function(table,key)
return Window.prototype[key]
end
function Window.new(t)
setmetatable(t,Window.mt)
return t
end
--测试
w = Window.new{x = 10,y = 20}
print(w.heigth)
__newindex的用法基本与例子2一致,区别在于__newindex的作用是用于table的更新,__index用于table的查询操作。
当对table中不存在的索引赋值时,就会调用__newindex元方法。组合使用__index和__newindex可以实现很多功能
封装:
有了元表(metatable)之后,Lua类的封装就会变得很容易。只要为table添加metatable,并设置__index方法。
例如:
People = {age = 18}
function People:new()
local p = {}
setmetatable(p,self)
self.__index = self
return p
end
function People:groupUp()
self.age = self.age + 1
print(self.age)
end
--测试
p1 = People:new()
p1:growUp() --19
p2 = People:new()
p2:growUp() --19
--[[运行结果可以看出:两个对象拥有的age成员是完全独立的,而且所有的有关People的方法都可以对外不可见,这样完全实现了面向对象中的类的封装--]]
继承:
继承是面向对象中必不可少的一部分,依然使用上例中的People,展示Lua中实现继承的方法。创建一个People实例Man,再在Man上重写People的同名方法:
Man = People:new()
function Man:growUp()
self.age = self.age + 1
print("man‘s growUp:"..self.age)
end
--测试
man1 = Man:new()
man1.growUp() --man‘s growUp:19
多态:
Lua不支持函数多态,而支持指针的多态,由于Lua动态类型的特性,本身就能支持。
person = People:new()
person:growUp() --People‘s growUp:19
person = Man:new()
person:growUp() --Man‘s growUp:19

cocos2d-Lua02Lua面向对象

标签:

原文地址:http://blog.csdn.net/opera95/article/details/51082322

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!