成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Lua面向對象編程的基本原理示例,你明白幾分?

開發 前端
這篇文章主要描述:在Lua?語言中,如何通過table結構來實現面向對象編程。主要是看到某鳥教程上錯誤百出,估計示例代碼自己都沒有測試過;關于Lua?語言中的table?以及metatable的基本知識,這里就不贅述了,官方手冊中描述的很清楚。

一些廢話

Lua語言是一個小而美的語言,使用者不多。

估計閱讀這篇文章的人也不會多,姑且當做一篇筆記吧。

這篇文章主要描述:在Lua語言中,如何通過table結構來實現面向對象編程。

主要是看到某鳥教程上錯誤百出,估計示例代碼自己都沒有測試過;

關于Lua語言中的table以及metatable的基本知識,這里就不贅述了,官方手冊中描述的很清楚。

測試代碼

1 #!/usr/bin/lua
2
3 ------------------------------ class A
4 A = {
5 a = 1,
6 funcA = function()
7 print("this is funcA")
8 end
9 }
10
11 function A:new(t)
12 local t = t or {}
13 self.__index = self
14 setmetatable(t, self)
15 return t
16 end
17
18 function A:myadd(num)
19 self.a = self.a + num
20 end
21
22 objA = A:new()
23 print("objA.a = " .. objA.a)
24 print(objA.funcA())
25 print(string.rep("-", 20))
26
27 ------------------------------ class B
28 B = A:new({
29 b = 2,
30 funcB = function()
31 print("this is funcB")
32 end
33 })
34
35 objB = B:new()
36 print("objB.a = " .. objB.a)
37 print("objB.b = " .. objB.b)
38 print(objB.funcA())
39 print(objB.funcB())
40
41 objB:myadd(10)
42 print("objA.a = " .. objA.a)
43 print("objB.a = " .. objB.a)

執行結果如下:

$ ./oop.lua 
objA.a = 1
this is funcA

--------------------
objB.a = 1
objB.b = 2
this is funcA

this is funcB

objA.a = 1
objB.a = 11

代碼說明

基類(父類) A

首先來分析下4-25行的代碼。

4-9行:定義父類A的成員變量和函數(按照C++中的習慣,可以叫做方法),可以看出Lua語言中的函數是“一等公民”,是可以賦值給一個變量的。

11-16行:相當于是構造函數,用來創建一個父類A的對象。

18-20行:給父類A增加一個函數,待會在分析子類B的時候再說。

22行:調用A:new()函數,創建一個類A的對象,賦值給變量objA。

在A:new()函數中,關鍵是第13行代碼:此時self等于A,就相當于是A.__index = A,這是合法的。

因為函數的調用方式是A:new(),Lua的語法糖會把A作為第一個參數傳遞給new()函數的第一個隱藏參數self。

然后執行14行的setmetatable(t, self),相當于把表t的元表設置為A。

以上兩行搞明白之后,23-24行的打印語句就簡單了:

23行:因為表objA中沒有成員a,但是objA被設置了元表A,而且該元表A帶有__index屬性,該屬性的值是表A自己,于是就到A中查找是否有成員a,于是就打印出:

objA.a = 1

__index 屬性的值,可以是一個表,可以是一個函數;

只不過這里特殊一點:__index 設置為 A 自己;

24行:查找函數的過程是一樣的,找到元表A的__index屬性的值,也就是表A自己中的funcA函數,然后調用,打印出:

this is funcA

派生類(子類) B

28-33行:定義了子類B,其實它也是一個對象。

在創建函數A:new(t)中,參數t的值是:

local t = {
b = 2,
funcB = function()
print("this is funcB")
end
}

此時,self仍然是父類A,B的創建過程與objA的創建過程是一樣的,只不過給參數t設置了子類B自己的成員變量和函數。

所以,B的元表被設置為A(14行代碼的功勞),當然了A的__index仍然被設置為A自己。

關鍵是35行:objB = B:new(),得仔細嘮嘮。

子類B并沒有自己的new函數,但是類B(也是一個 table) 的元表被設置為A,并且A.__index = A,所以最終就找到了A中的new函數,也就是11-16行代碼。

進入這個函數中時,第一個隱藏參數self被設置為 B 了,因為函數調用形式是:B:new()。

所以:

  • 13 行 self.__index = self  相當于設置 B.__index = B
  • 14 行 etmetatable(t, self) 相當于把表 t 的元表設置為 B

new()函數返回之后,就把t賦值給objB。

下面再看一下36-39行的打印語句:

36 print("objB.a = " .. objB.a)
37 print("objB.b = " .. objB.b)
38 print(objB.funcA())
39 print(objB.funcB())

36行:objB中并沒有成員a,但是objB的元表是B,而且B.__index = B,所以就到B中去查找a。

雖然B中也沒有a,但是B的元表是A,而且A.__index = A,所以就在A中找到了成員a,打印出:

objB.a = 1

37行:objB中并沒有成員b,但是objB的元表是B,而且B.__index = B,所以在B中找到了成員b,因此打印出:

objB.b = 2

37和38行的查找過程是類似的,只不過換成了函數而已。

子類對象操作自己的變量

41行:objB:myadd(10)。

查找myadd函數的過程與查找obj.a的過程是一樣的,這里再嘮叨一遍:

  • objB 中并沒有函數 myadd,但是 objB 的元表是 B,而且 B.__index = B,所以就到 B 中去查找 myadd;
  • 雖然 B 中也沒有 myadd,但是 B 的元表是 A,而且 A.__index = A,所以就在 A 中找到了函數 myadd;

于是就調用了函數:

18 function A:myadd(num)
19 self.a = self.a + num
20 end

而且self等于objB,因此函數體中就等于是:

objB.a = objB.a + 10

加法表達式中的objB.a的讀取過程,上面已經描述過了,最終定位到的是父類A中的a,即:1。

1 + 10 = 11,然后把11賦值給objB.a。

在賦值操作中,被賦值的objB.a就不再是父類A中的那個a了!

因為objB本質是一個table,給objB設置鍵值對的時候:

  • 如果鍵已經存在了,那么就直接設置該鍵的值;
  • 如果鍵不存在,那么 lua 會看它的元表中是否有 __newindex 字段(可以是一個table,也可以是一個函數);
  • 2-1. 如果有   __newindex 字段,那么就是調用 __newindex (如果是一個函數),或者在 __newindex 中添加鍵值對(如果是一個table);
  • 2-2. 如果沒有 __newindex 字段,那么就直接在 objB 中存儲該鍵值對;

根據上面這個規則,就會設置objB.a = 11。

明白以上這些之后,42和43行的打印語句就不復雜了。

42行:objA最終找到的a是父類A中的成員a,打印出:objA.a = 1。

43行:objB中自己已經有了成員a,所以打印出:objB.a = 11。

繼續往下繼承

有了上面的基礎,再從子類B中派生出類C,C派生出類D... 都不是什么問題了,如下所示:

C = B:new()
objC = C:new()
print("objC.a = " .. objC.a)
print("objC.b = " .. objC.b)
print(objC.funcA())
print(objC.funcB())

感興趣的讀者可以自己測試一下。

責任編輯:武曉燕 來源: IOT物聯網小鎮
相關推薦

2024-01-30 09:21:01

deltaFifo機制apiServer

2019-11-28 10:45:28

ZooKeeper源碼分布式

2012-01-12 14:37:34

jQuery

2010-01-07 09:53:09

Winform多線程編

2010-08-20 13:29:33

OFDM

2013-04-07 14:09:55

Android應用基本

2020-03-21 14:57:14

手機定位智能手機APP

2009-02-24 09:43:00

IP電話原理

2011-11-29 12:17:00

2010-09-08 15:25:15

Linux系統分區

2011-08-10 19:33:09

Cocoa對象

2010-03-17 13:35:02

2016-08-18 00:04:09

網絡爬蟲抓取系統服務器

2016-08-17 23:53:29

網絡爬蟲抓取系統

2021-02-08 21:40:04

SockmapBPF存儲

2011-07-07 14:22:27

Cocoa 對象 生命周期

2010-06-18 17:28:37

Linux Anacr

2013-01-09 09:27:09

面向對象計算機編程

2020-11-26 13:54:03

容器LinuxDocker

2011-07-07 14:10:21

Cocoa 內省 hash
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: av毛片在线播放 | 一级h片 | 国产午夜高清 | 免费精品在线视频 | 91成人影院 | 一区二区视频在线 | 日韩www| 亚洲a视频| 国产三级在线观看播放 | 欧洲一级毛片 | 91国产精品在线 | 国产高清一区二区三区 | 成人福利在线 | 国产成人亚洲精品自产在线 | 一级黄色片日本 | 999精品视频 | 日韩a | 免费观看一级毛片 | 欧美99| 视频一区二区三区四区五区 | 国产精品一区二区三区久久久 | 91视频在线看 | 国产在线永久免费 | 色婷婷亚洲国产女人的天堂 | 日韩久久网 | 久久久久久亚洲精品 | 久久精品无码一区二区三区 | 日韩精品国产精品 | 成人欧美日韩一区二区三区 | 欧美黄色一区 | 国产精品99久久免费观看 | 看真人视频一级毛片 | 欧美在线成人影院 | 亚洲一区 | 国产目拍亚洲精品99久久精品 | 黄色一级毛片 | 国内精品伊人久久久久网站 | 亚洲影视在线 | 亚洲综合首页 | 狠狠躁夜夜躁人人爽天天高潮 | 亚洲 91|