# Lua面向对象编程
#### 类
在 Lua 中,我们可以使用表和函数实现面向对象。将函数和相关的数据放置于同一个表中就形成了一个对象。
~~~
Account = {balance = 0}
function Account:deposit (v) --注意,此处使用冒号,可以免写self关键字;如果使用.号,第一个参数必须是self
self.balance = self.balance + v
end
function Account:withdraw (v) --注意,此处使用冒号,可以免写self关键字;
if self.balance > v then
self.balance = self.balance - v
else
error("insufficient funds")
end
end
function Account:new (o) --注意,此处使用冒号,可以免写self关键字;
o = o or {} -- create object if user does not provide one
setmetatable(o, {__index = self})
return o
end
a = Account:new()
a:deposit(100)
b = Account:new()
b:deposit(50)
print(a.balance) -->100
print(b.balance) -->50
--本来笔者开始是自己写的例子,但发现的确不如Lua作者给的例子经典,所以还是沿用作者的代码。
~~~
上面这段代码"setmetatable(o, {__index = self})"这句话值得注意。根据我们在元表这一章学到的知识,我们明白,setmetatable将Account作为新建'o'表的原型,所以当o在自己的表内找不到'balance'、'withdraw'这些方法和变量的时候,便会到__index所指定的Account类型中去寻找。
#### 继承
继承可以用元表实现,它提供了在父类中查找存在的方法和变量的机制。
~~~
--定义继承
--定义继承
SpecialAccount = Account:new({limit = 1000}) --开启一个特殊账户类型,这个类型的账户可以取款超过余额限制1000元
function SpecialAccount:withdraw (v)
if v - self.balance >= self:getLimit() then
error("insufficient funds")
end
self.balance = self.balance - v
end
function SpecialAccount:getLimit ()
return self.limit or 0
end
spacc = SpecialAccount:new()
spacc:withdraw(100)
print(spacc.balance) --> -100
acc = Account:new()
acc:withdraw(100) --> 超出账户余额限制,抛出一个错误
~~~
#### 多重继承
多重继承肯定不能采用我们在单继承中的所使用的方法,因为直接采用setmetatable的方式,会造成metatable的覆盖。在多重继承中,我们自己利用'__index'元方法定义恰当的访问行为。
~~~
local function search (k, plist)
for i=1, table.getn(plist) do
local v = plist[i][k] -- try 'i'-th superclass
if v then return v end
end
end
function createClass (...)
local c = {} -- new class
-- class will search for each method in the list of its
-- parents (`args' is the list of parents)
args = {...}
setmetatable(c, {__index = function (self, k)
return search(k, args)
end})
-- prepare `c' to be the metatable of its instances
c.__index = c
-- define a new constructor for this new class
function c:new (o)
o = o or {}
setmetatable(o, c)
return o
end
-- return new class
return c
end
~~~
解释一下上面的代码。我们定义了一个通用的创建多重继承类的函数'createClass',这个函数可以接受多个类。如何让我们新建的多重继承类恰当地访问从不同类中继承来的函数或者成员变量呢?我们就用到了'search'函数,该函数接受两个参数,第一个参数是想要访问的类成员的名字,第二个参数是被继承的类列表。通过一个for循环在列表的各个类中寻找想要访问成员。
我们再定一个新类,来验证'createClass'的正确性。
~~~
Named = {}
function Named:getname ()
return self.name
end
function Named:setname (n)
self.name = n
end
NamedAccount = createClass(Account, Named) --同时继承Account 和 Named两个类
account = NamedAccount:new{name = "Paul"} --使用这个多重继承类定义一个实例
print(account:getname()) --> Pauls
account:deposit(100)
print(account.balance) --> 100
~~~
#### 成员私有性
在面向对象当中,如何将成员内部实现细节对使用者隐藏,也是值得关注的一点。在 Lua 中,成员的私有性,使用类似于函数闭包的形式来实现。在我们之前的银行账户的例子中,我们使用一个工厂方法来创建新的账户实例,通过工厂方法对外提供的闭包来暴露对外接口。而不想暴露在外的例如balace成员变量,则被很好的隐藏起来。
~~~
function newAccount (initialBalance)
local self = {balance = initialBalance}
local withdraw = function (v)
self.balance = self.balance - v
end
local deposit = function (v)
self.balance = self.balance + v
end
local getBalance = function () return self.balance end
return {
withdraw = withdraw,
deposit = deposit,
getBalance = getBalance
}
end
a = newAccount(100)
a.deposit(100)
print(a.getBalance()) --> 200
print(a.balance) --> nil
~~~
- 序
- Lua 入门
- Lua简介
- Lua环境搭建
- 基础数据类型
- 表达式
- 控制结构
- if/else
- while
- repeat
- for
- break,return
- Lua函数
- 函数的定义
- 函数的参数
- 函数的返回值
- 函数回调
- 模块
- String库
- Table库
- 日期时间函数
- 数学库函数
- 文件操作
- 元表
- 面向对象编程
- FFI
- 下标从1开始
- 局部变量
- 判断数组大小
- 非空判断
- 正则表达式
- 不用标准库
- 虚变量
- 函数在调用代码前定义
- 抵制使用module()函数来定义Lua模块
- 点号与冒号操作符的区别
- Nginx
- Nginx 新手起步
- location 匹配规则
- if 是邪恶的
- 静态文件服务
- 日志服务
- 反向代理
- 负载均衡
- 陷阱和常见错误
- 环境搭建
- Windows平台
- CentOS平台
- Ubuntu平台
- Mac OS X平台
- Hello World
- 简单API Server框架
- 获取Nginx内置绑定变量
- LuaRestyRedisLibrary
- select+set_keepalive组合操作引起的数据读写错误
- redis接口的二次封装(简化建连、拆连等细节)
- redis接口的二次封装(发布订阅)
- pipeline压缩请求数量
- script压缩复杂请求
- LuaCjsonLibrary
- json解析的异常捕获
- 稀疏数组
- 空table编码为array还是object
- 跨平台的库选择
- PostgresNginxModule
- 调用方式简介
- 不支持事务
- 超时
- 健康监测
- SQL注入
- LuaNginxModule
- 执行阶段概念
- 正确的记录日志
- 热装载代码
- 阻塞操作
- 缓存
- sleep
- 定时任务
- 禁止某些终端访问
- 请求返回后继续执行
- 调试
- 调用其他C函数动态库
- 我的lua代码需要调优么
- 变量的共享范围
- 动态限速
- shared.dict 非队列性质
- 如何添加自己的lua api
- 正确使用长链接
- 如何引用第三方resty库
- 典型应用场景
- LuaRestyDNSLibrary
- 使用动态DNS来完成HTTP请求
- 缓存失效风暴
- 测试
- 单元测试
- API测试
- 性能测试
- 持续集成
- 灰度发布
- Web 服务
- API的设计
- 数据合法性检测
- 协议无痛升级
- 代码规范
- 连接池
- C10K编程
- TIME_WAIT问题
- 与Docker使用的网络瓶颈
- 火焰图
- 什么时候使用
- 显示的是什么
- 如何安装火焰图生成工具
- 如何定位问题
- 开源文化对360企业安全的影响