Step By Step(Lua面向对象)

Lua中的table就是一种对象,但是如果直接使用仍然会存在大量的问题,见如下代码:

  1. 1 Account = {balance = 0}
    2 function Account.withdraw(v)
    3 Account.balance = Account.balance - v
    4 end
    5 --下面是测试调用函数
    6 Account.withdraw(100.00)

在上面的withdraw函数内部依赖了全局变量Account,一旦该变量发生改变,将会导致withdraw不再能正常的工作,如:

  1. 1 a = Account; Account = nil
    2 a.withdraw(100.00) --将会导致访问空nil的错误。

这种行为明显的违反了面向对象封装性和实例独立性。要解决这一问题,我们需要给withdraw函数在添加一个参数self,他等价于Java/C++中的this,见如下修改:

  1. 1 function Account.withdraw(self,v)
    2 self.balance = self.balance - v
    3 end
    4 --下面是基于修改后代码的调用:
    5 a1 = Account; Account = nil
    6 a1.withdraw(a1,100.00) --正常工作。

针对上述问题,Lua提供了一种更为便利的语法,即将点(.)替换为冒号(:),这样可以在定义和调用时均隐藏self参数,如:

  1. 1 function Account:withdraw(v)
    2 self.balance = self.balance - v
    3 end
    4 --调用代码可改为:
    5 a:withdraw(100.00)

1. 类:
    Lua在语言上并没有提供面向对象的支持,因此想实现该功能,我们只能通过table来模拟,见如下代码及关键性注释:

 
  1. 1 --[[
    2 在这段代码中,我们可以将Account视为class的声明,如Java中的:
    3 public class Account
    4 {
    5 public float balance = 0;
    6 public Account(Account o);
    7 public void deposite(float f);
    8 }
    9 --]]
    10 --这里balance是一个公有的成员变量。
    11 Account = {balance = 0}
    12
    13 --new可以视为构造函数
    14 function Account:new(o)
    15 o = o or {} --如果参数中没有提供table,则创建一个空的。
    16 --将新对象实例的metatable指向Account表(类),这样就可以将其视为模板了。
    17 setmetatable(o,self)
    18 --在将Account__index字段指向自己,以便新对象在访问Account的函数和字段时,可被直接重定向。
    19 self.__index = self
    20 --最后返回构造后的对象实例
    21 return o
    22 end
    23
    24 --deposite被视为Account类的公有成员函数
    25 function Account:deposit(v)
    26 --这里的self表示对象实例本身
    27 self.balance = self.balance + v
    28 end
    29
    30 --下面的代码创建两个Account的对象实例
    31
    32 --通过Accountnew方法构造基于该类的示例对象。
    33 a = Account:new()
    34 --[[
    35 这里需要具体解释一下,此时由于table a中并没有deposite字段,因此需要重定向到Account
    36 同时调用Accountdeposite方法。在Account.deposite方法中,由于self(a对象)并没有balance
    37 字段,因此在执行self.balance + v时,也需要重定向访问Account中的balance字段,其缺省值为0
    38 在得到计算结果后,再将该结果直接赋值给a.balance。此后a对象就拥有了自己的balance字段和值。
    39 下次再调用该方法,balance字段的值将完全来自于a对象,而无需在重定向到Account了。
    40 --]]
    41 a:deposit(100.00)
    42 print(a.balance) --输出100
    43
    44 b = Account:new()
    45 b:deposit(200.00)
    46 print(b.balance) --输出200
 

2. 继承:
    继承也是面向对象中一个非常重要的概念,在Lua中我们仍然可以像模拟类那样来进一步实现面向对象中的继承机制,见如下代码及关键性注释:

 
  1. 1 --需要说明的是,这段代码仅提供和继承相关的注释,和类相关的注释在上面的代码中已经给出。
    2 Account = {balance = 0}
    3
    4 function Account:new(o)
    5 o = o or {}
    6 setmetatable(o,self)
    7 self.__index = self
    8 return o
    9 end
    10
    11 function Account:deposit(v)
    12 self.balance = self.balance + v
    13 end
    14
    15 function Account:withdraw(v)
    16 if v > self.balance then
    17 error("Insufficient funds")
    18 end
    19 self.balance = self.balance - v
    20 end
    21
    22 --下面将派生出一个Account的子类,以使客户可以实现透支的功能。
    23 SpecialAccount = Account:new() --此时SpecialAccount仍然为Account的一个对象实例
    24
    25 --派生类SpecialAccount扩展出的方法。
    26 --下面这些SpecialAccount中的方法代码(getLimit/withdraw),一定要位于SpecialAccountAccount构造之后。
    27 function SpecialAccount:getLimit()
    28 --此时的self将为对象实例。
    29 return self.limit or 0
    30 end
    31
    32 --SpecialAccount将为Account的子类,下面的方法withdraw可以视为SpecialAccount
    33 --重写的Account中的withdraw方法,以实现自定义的功能。
    34 function SpecialAccount:withdraw(v)
    35 --此时的self将为对象实例。
    36 if v - self.balance >= self:getLimit() then
    37 error("Insufficient funds")
    38 end
    39 self.balance = self.balance - v
    40 end
    41
    42 --在执行下面的new方法时,table s的元表已经是SpecialAccount了,而不再是Account
    43 s = SpecialAccount:new{limit = 1000.00}
    44 --在调用下面的deposit方法时,由于table sSpecialAccount均未提供该方法,因此访问的仍然是
    45 --Accountdeposit方法。
    46 s:deposit(100)
    47
    48
    49 --此时的withdraw方法将不再是Account中的withdraw方法,而是SpecialAccount中的该方法。
    50 --这是因为Lua先在SpecialAccount(即s的元表)中找到了该方法。
    51 s:withdraw(200.00)
    52 print(s.balance) --输出-100
 

3. 私密性:
    私密性对于面向对象语言来说是不可或缺的,否则将直接破坏对象的封装性。Lua作为一种面向过程的脚本语言,更是没有提供这样的功能,然而和模拟支持类与继承一样,我们仍然可以在Lua中通过特殊的编程技巧来实现它,这里我们应用的是Lua中的闭包函数。该实现方式和前面两个示例中基于元表的方式有着很大的区别,见如下代码示例和关键性注释:

 
  1. 1 --这里我们需要一个闭包函数作为类的创建工厂
    2 function newAccount(initialBalance)
    3 --这里的self仅仅是一个普通的局部变量,其含义完全不同于前面示例中的self
    4 --这里之所以使用self作为局部变量名,也是为了方便今后的移植。比如,以后
    5 --如果改为上面的实现方式,这里应用了self就可以降低修改的工作量了。
    6 local self = {balance = initialBalance} --这里我们可以将self视为私有成员变量
    7 local withdraw = function(v) self.balance = self.balance - v end
    8 local deposit = function(v) self.balance = self.balance + v end
    9 local getBalance = function() return self.balance end
    10 --返回对象中包含的字段仅仅为公有方法。事实上,我们通过该种方式,不仅可以实现
    11 --成员变量的私有性,也可以实现方法的私有性,如:
    12 --local privateFunction = function() --do something end
    13 --只要我们不在输出对象中包含该方法的字段即可。
    14 return {withdraw = withdraw, deposit = deposit, getBalance = getBalance}
    15 end
    16
    17 --和前面两个示例不同的是,在调用对象方法时,不再需要self变量,因此我们可以直接使用点(.),
    18 --而不再需要使用冒号(:)操作符了。
    19 accl = newAccount(100.00)
    20 --在函数newAccount返回之后,该函数内的“非局部变量”表self就不再能被外部访问了,只能通过
    21 --该函数返回的对象的方法来操作它们。
    22 accl.withdraw(40.00)
    23 print(acc1.getBalance())
 

事实上,上面的代码只是给出一个简单的示例,在实际应用中,我们可以将更多的私有变量存放于上例的局部self表中。

 
 
 

Step By Step(Lua面向对象)的更多相关文章

  1. Step By Step(Lua开篇)

    Step By Step(Lua开篇) 一.简介: Lua作为目前最为流行的.免费轻量级嵌入式脚本语言,在很多工业级的应用程序中被广泛应用,如Adobe's Photoshop,甚至是在一些著名的游戏 ...

  2. Step By Step(C调用Lua)

    Step By Step(C调用Lua) 1. 基础:    Lua的一项重要用途就是作为一种配置语言.现在从一个简单的示例开始吧.    --这里是用Lua代码定义的窗口大小的配置信息    wid ...

  3. Step By Step(Lua系统库)

    Step By Step(Lua系统库) Lua为了保证高度的可移植性,因此,它的标准库仅仅提供了非常少的功能,特别是和OS相关的库.但是Lua还提供了一些扩展库,比如Posix库等.对于文件操作而言 ...

  4. Step By Step(Lua输入输出库)

    Step By Step(Lua输入输出库) I/O库为文件操作提供了两种不同的模型,简单模型和完整模型.简单模型假设一个当前输入文件和一个当前输出文件,他的I/O操作均作用于这些文件.完整模型则使用 ...

  5. Step By Step(Lua字符串库)

    Step By Step(Lua字符串库) 1. 基础字符串函数:    字符串库中有一些函数非常简单,如:    1). string.len(s) 返回字符串s的长度:    2). string ...

  6. Step By Step(Lua弱引用table)

    Step By Step(Lua弱引用table) Lua采用了基于垃圾收集的内存管理机制,因此对于程序员来说,在很多时候内存问题都将不再困扰他们.然而任何垃圾收集器都不是万能的,在有些特殊情况下,垃 ...

  7. Step By Step(Lua模块与包)

    Step By Step(Lua模块与包) 从Lua 5.1开始,我们可以使用require和module函数来获取和创建Lua中的模块.从使用者的角度来看,一个模块就是一个程序库,可以通过requi ...

  8. Step By Step(Lua环境)

    Step By Step(Lua环境) Lua将其所有的全局变量保存在一个常规的table中,这个table被称为"环境".它被保存在全局变量_G中.    1. 全局变量声明:  ...

  9. Step By Step(Lua元表与元方法)

    Step By Step(Lua元表与元方法) Lua中提供的元表是用于帮助Lua数据变量完成某些非预定义功能的个性化行为,如两个table的相加.假设a和b都是table,通过元表可以定义如何计算表 ...

随机推荐

  1. ThnikPHP3.2 学习链接整理

    ThnikPHP3.2 学习链接整理 ThinkPHP3.2.3 U()方法的使用总结 看云手册 ThinkPHP3.2完全开发手册 TP3.2单字母函数 TP3.x中 M方法和D方法的区别

  2. POJ2983 查分约束系统

    题意:        给你n个点,然后给你两种情况,P a b c,表明a在b的北边c那么远,V a b 表明a在b的北边(距离最少是1),问你这些条件是否冲突. 思路:       一开始想用带权并 ...

  3. 使用BurpSuite抓取HTTPS网站的数据包

    昨天面试,技术官问到了我如何使用BurpSuite抓取https网站的数据包,一时间没能回答上来(尴尬!).因为以前https网站的数据包我都是用Fiddler抓取的,Fiddlert自动帮我们配置好 ...

  4. 学习Canvas绘图与动画基础 canvas入门(一)

    一.创建canvas 1 <!DOCTYPE html> 2 <html> 3 <head lang="en"> 4 <meta char ...

  5. XCTF-supersqli

    supersqli 进来有个输入框,看内容应该是var_dump了sql查询结果 单引号有报错,万能语句能用,注释符#没被ban 打了个union select,给提示ban了一堆关键字,而且忽略大小 ...

  6. 【JavaScript】Leetcode每日一题-在D天内送包裹的能力

    [JavaScript]Leetcode每日一题-在D天内送包裹的能力 [题目描述] 传送带上的包裹必须在 D 天内从一个港口运送到另一个港口. 传送带上的第 i 个包裹的重量为 weights[i] ...

  7. vscode 超好用的前端插件

    一 vscode 前端调试接口的插件 作为前端工程师,接口的调试是我们必不可少的工作.以前用过postman,但是作为一个vscode重度使用者,我希望看看vscode能否进行对接口的调试.省的跟后台 ...

  8. IOC随笔小记录

    对IOC的一点学习笔记 IOC (Inversion of Control):控制反转 DI (Dependency Injection):依赖注入 1.在没有使用IOC的情况下是如何进行的 在Use ...

  9. python工业互联网应用实战15-前后端分离模式1

    我们在13章节里通过监控界面讲了如何使用jquery的动态加载数据写法,通过简单案例来说明了如何实现动态的刷新监控界面的数据,本章我们将演示如何从Django模板加载数据逐步演化到前后端分离的异步数据 ...

  10. 使用小记:Zookeeper中动态改变节点的功能

    Zookeeper 3.5+提供了reconfig功能实现动态配置节点,官方的说明是,"你再也不需要进行全部节点重启"就可以应用所有的修改: http://zookeeper.ap ...