Lua 提供了 debug 庫(kù)用于提供創(chuàng)建我們自定義調(diào)速器的功能。Lua 本身并未有內(nèi)置的調(diào)速器,但很多開發(fā)者共享了他們的 Lua 調(diào)速器代碼。
Lua 中 debug 庫(kù)包含以下函數(shù):
sethook ([thread,] hook, mask [, count]):序號(hào) | 方法 & 用途 |
---|---|
1. | debug(): 進(jìn)入一個(gè)用戶交互模式,運(yùn)行用戶輸入的每個(gè)字符串。 使用簡(jiǎn)單的命令以及其它調(diào)試設(shè)置,用戶可以檢閱全局變量和局部變量, 改變變量的值,計(jì)算一些表達(dá)式,等等。 |
2. | getfenv(object): 返回對(duì)象的環(huán)境變量。 |
3. | gethook(optional thread): 返回三個(gè)表示線程鉤子設(shè)置的值: 當(dāng)前鉤子函數(shù),當(dāng)前鉤子掩碼,當(dāng)前鉤子計(jì)數(shù) |
4. | getinfo ([thread,] f [, what]): 返回關(guān)于一個(gè)函數(shù)信息的表。 你可以直接提供該函數(shù), 也可以用一個(gè)數(shù)字 f 表示該函數(shù)。 數(shù)字 f 表示運(yùn)行在指定線程的調(diào)用棧對(duì)應(yīng)層次上的函數(shù): 0 層表示當(dāng)前函數(shù)(getinfo 自身); 1 層表示調(diào)用 getinfo 的函數(shù) (除非是尾調(diào)用,這種情況不計(jì)入棧);等等。 如果 f 是一個(gè)比活動(dòng)函數(shù)數(shù)量還大的數(shù)字, getinfo 返回 nil。 |
5. | debug.getlocal ([thread,] f, local): 此函數(shù)返回在棧的 f 層處函數(shù)的索引為 local 的局部變量 的名字和值。 這個(gè)函數(shù)不僅用于訪問顯式定義的局部變量,也包括形參、臨時(shí)變量等。 |
6. | getmetatable(value): 把給定索引指向的值的元表壓入堆棧。如果索引無效,或是這個(gè)值沒有元表,函數(shù)將返回 0 并且不會(huì)向棧上壓任何東西。 |
7. | getregistry(): 返回注冊(cè)表表,這是一個(gè)預(yù)定義出來的表, 可以用來保存任何 C 代碼想保存的 Lua 值。 |
8. | getupvalue (f, up) 此函數(shù)返回函數(shù) f 的第 up 個(gè)上值的名字和值。 如果該函數(shù)沒有那個(gè)上值,返回 nil 。 |
10. | 將一個(gè)函數(shù)作為鉤子函數(shù)設(shè)入。 字符串 mask 以及數(shù)字 count 決定了鉤子將在何時(shí)調(diào)用。 掩碼是由下列字符組合成的字符串,每個(gè)字符有其含義:
|
11. | setlocal ([thread,] level, local, value): 這個(gè)函數(shù)將 value 賦給 棧上第 level 層函數(shù)的第 local 個(gè)局部變量。 如果沒有那個(gè)變量,函數(shù)返回 nil 。 如果 level 越界,拋出一個(gè)錯(cuò)誤。 |
12. | setmetatable (value, table): 將 value 的元表設(shè)為 table (可以是 nil)。 返回 value。 |
13. | setupvalue (f, up, value): 這個(gè)函數(shù)將 value 設(shè)為函數(shù) f 的第 up 個(gè)上值。 如果函數(shù)沒有那個(gè)上值,返回 nil 否則,返回該上值的名字。 |
14. | traceback ([thread,] [message [, level]]): 如果 message 有,且不是字符串或 nil, 函數(shù)不做任何處理直接返回 message。 否則,它返回調(diào)用棧的?;厮菪畔?。 字符串可選項(xiàng) message 被添加在?;厮菪畔⒌拈_頭。 數(shù)字可選項(xiàng) level 指明從棧的哪一層開始回溯 (默認(rèn)為 1 ,即調(diào)用 traceback 的那里)。 |
上表列出了我們常用的調(diào)試函數(shù),接下來我們可以看些簡(jiǎn)單的例子:
function myfunction () print(debug.traceback("Stack trace")) print(debug.getinfo(1)) print("Stack trace end") return 10 end myfunction () print(debug.getinfo(1))
執(zhí)行以上代碼輸出結(jié)果為:
Stack trace stack traceback: test2.lua:2: in function 'myfunction' test2.lua:8: in main chunk [C]: ? table: 0054C6C8 Stack trace end
在以實(shí)例中,我們使用到了 debug 庫(kù)的 traceback 和 getinfo 函數(shù), getinfo 函數(shù)用于返回函數(shù)信息的表。
我們經(jīng)常需要調(diào)試函數(shù)的內(nèi)的局部變量。我們可以使用 getupvalue 函數(shù)來設(shè)置這些局部變量。實(shí)例如下:
function newCounter () local n = 0 local k = 0 return function () k = n n = n + 1 return n end end counter = newCounter () print(counter()) print(counter()) local i = 1 repeat name, val = debug.getupvalue(counter, i) if name then print ("index", i, name, "=", val) if(name == "n") then debug.setupvalue (counter,2,10) end i = i + 1 end -- if until not name print(counter())
執(zhí)行以上代碼輸出結(jié)果為:
1 2 index 1 k = 1 index 2 n = 2 11
在以上實(shí)例中,計(jì)數(shù)器在每次調(diào)用時(shí)都會(huì)自增1。實(shí)例中我們使用了 getupvalue 函數(shù)查看局部變量的當(dāng)前狀態(tài)。我們可以設(shè)置局部變量為新值。實(shí)例中,在設(shè)置前 n 的值為 2,使用 setupvalue 函數(shù)將其設(shè)置為 10?,F(xiàn)在我們調(diào)用函數(shù),執(zhí)行后輸出為 11 而不是 3。
命令行調(diào)試器有:RemDebug、clidebugger、ctrace、xdbLua、LuaInterface - Debugger、Rldb、ModDebug。
圖形界調(diào)試器有:SciTE、Decoda、ZeroBrane Studio、akdebugger、luaedit。
更多建議: