Lua 調(diào)試(Debug)

2022-06-25 15:20 更新

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á)式,等等。
輸入一行僅包含 cont 的字符串將結(jié)束這個(gè)函數(shù), 這樣調(diào)用者就可以繼續(xù)向下運(yùn)行。

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 。
以 '(' (開括號(hào))打頭的變量名表示沒有名字的變量 (去除了調(diào)試信息的代碼塊)。

10.將一個(gè)函數(shù)作為鉤子函數(shù)設(shè)入。 字符串 mask 以及數(shù)字 count 決定了鉤子將在何時(shí)調(diào)用。 掩碼是由下列字符組合成的字符串,每個(gè)字符有其含義:

  • 'c': 每當(dāng) Lua 調(diào)用一個(gè)函數(shù)時(shí),調(diào)用鉤子;
  • 'r': 每當(dāng) Lua 從一個(gè)函數(shù)內(nèi)返回時(shí),調(diào)用鉤子;
  • 'l': 每當(dāng) Lua 進(jìn)入新的一行時(shí),調(diào)用鉤子。
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ù)信息的表。

另一個(gè)實(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)試類型

  • 命令行調(diào)試
  • 圖形界面調(diào)試

命令行調(diào)試器有:RemDebug、clidebugger、ctrace、xdbLua、LuaInterface - Debugger、Rldb、ModDebug。

圖形界調(diào)試器有:SciTE、Decoda、ZeroBrane Studio、akdebugger、luaedit。


以上內(nèi)容是否對(duì)您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號(hào)
微信公眾號(hào)

編程獅公眾號(hào)