【Lua从青铜到王者基础篇】第十二篇:Lua错误处理

x33g5p2x  于2022-04-06 转载在 其他  
字(4.4k)|赞(0)|评价(0)|浏览(882)

系列文章目录

前言

🌲一、Lua 错误处理

程序运行中错误处理是必要的,在我们进行文件操作,数据转移及web service 调用过程中都会出现不可预期的错误。如果不注重错误信息的处理,就会造成信息泄露,程序无法运行等情况。

  • 任何程序语言中,都需要错误处理。错误类型有:
  • 语法错误。
  • 运行错误。

**简单模式在做一些简单的文件操作时较为合适。**但是在进行一些高级的文件操作的时候,简单模式就显得力不从心。例如同时读取多个文件这样的操作,使用完全模式则较为合适。

🌳二、语法错误

简单模式使用标准的 I/O 或使用一个当前输入文件和一个当前输出文件。

语法错误通常是由于对程序的组件(如运算符、表达式)使用不当引起的。一个简单的实例如下:

  1. -- test.lua 文件
  2. a == 2

以上代码的输出结果为:

  1. [{
  2. "resource": "/D:/vscode lua代码/hello world/Untitled-1.lua",
  3. "owner": "_generated_diagnostic_collection_name_#0",
  4. "code": "err-assign-as-eq",
  5. "severity": 8,
  6. "message": "应使用`=`来进行赋值操作。",
  7. "source": "Lua 语法检查",
  8. "startLineNumber": 1399,
  9. "startColumn": 2,
  10. "endLineNumber": 1399,
  11. "endColumn": 4
  12. }]

正如你所看到的,以上出现了语法错误,一个 “=” 号跟两个 “=” 号是有区别的。一个 “=” 是赋值表达式两个 “=” 是比较运算。另外一个实例:

  1. for a= 1,10
  2. print(a)
  3. end

以上代码的输出结果为:

  1. [{
  2. "resource": "/D:/vscode lua代码/hello world/Untitled-1.lua",
  3. "owner": "_generated_diagnostic_collection_name_#0",
  4. "code": "miss-symbol",
  5. "severity": 8,
  6. "message": "缺少符号`,`。",
  7. "source": "Lua 语法检查",
  8. "startLineNumber": 1401,
  9. "startColumn": 12,
  10. "endLineNumber": 1402,
  11. "endColumn": 4
  12. },{
  13. "resource": "/D:/vscode lua代码/hello world/Untitled-1.lua",
  14. "owner": "_generated_diagnostic_collection_name_#0",
  15. "code": "miss-symbol",
  16. "severity": 8,
  17. "message": "缺少符号`do`。",
  18. "source": "Lua 语法检查",
  19. "startLineNumber": 1402,
  20. "startColumn": 12,
  21. "endLineNumber": 1402,
  22. "endColumn": 12
  23. },{
  24. "resource": "/D:/vscode lua代码/hello world/Untitled-1.lua",
  25. "owner": "_generated_diagnostic_collection_name_#0",
  26. "code": "undefined-global",
  27. "severity": 4,
  28. "message": "未定义的全局变量 `a`。",
  29. "source": "Lua 诊断",
  30. "startLineNumber": 1402,
  31. "startColumn": 10,
  32. "endLineNumber": 1402,
  33. "endColumn": 11
  34. }]

语法错误比程序运行错误更简单,运行错误无法定位具体错误,而语法错误我们可以很快的解决,如以上实例我们只要在for语句下添加 do 即可:

  1. for a= 1,10
  2. do
  3. print(a)
  4. end

以上代码的输出结果为:

  1. 1
  2. 2
  3. 3
  4. 4
  5. 5
  6. 6
  7. 7
  8. 8
  9. 9
  10. 10

🌴三、运行错误

运行错误是程序可以正常执行,但是会输出报错信息。如下实例由于参数输入错误,程序执行时报错:

  1. function add(a,b)
  2. return a+b
  3. end
  4. add(10)

当我们编译运行以下代码时,编译是可以成功的,但在运行的时候会产生如下错误::

  1. c:/Users/Administrator/.vscode/extensions/actboy168.lua-debug-1.53.1/runtime/win32-x64/lua54/lua.exe: D:\vscode lua����\hello world/Untitled-1.lua:1409: attempt to perform arithmetic on a nil value (local 'b')
  2. stack traceback:
  3. D:\vscode lua����\hello world/Untitled-1.lua:1409: in function 'add'
  4. D:\vscode lua����\hello world/Untitled-1.lua:1412: in main chunk
  5. [C]: in ?

  • lua 里调用函数时,即使实参列表和形参列表不一致也能成功调用,多余的参数会被舍弃,缺少的参数会被补为 nil。
  • 以上报错信息是由于参数 b 被补为 nil 后,nil 参与了 + 运算。
  • 假如 add 函数内不是 “return a+b” 而是 “print(a,b)” 的话,结果会变成 “10 nil” 不会报错。

🌵四、错误处理

我们可以使用两个函数:assert 和 error 来处理错误。实例如下:

  1. local function add(a,b)
  2. assert(type(a) == "number", "a 不是一个数字")
  3. assert(type(b) == "number", "b 不是一个数字")
  4. return a+b
  5. end
  6. add(10)

实例中assert首先检查第一个参数,若没问题,assert不做任何事情;否则,assert以第二个参数作为错误信息抛出。

🌲1.error函数

语法格式:

  1. error (message [, level])

功能:终止正在执行的函数,并返回message的内容作为错误信息(error函数永远都不会返回),通常情况下,error会附加一些错误位置的信息到message头部。

  • Level参数指示获得错误的位置:
  • Level=1[默认]:为调用error位置(文件+行号)。
  • Level=2:指出哪个调用error的函数的函数。
  • Level=0:不添加错误位置信息。

🌳2.pcall函数

  • Lua中处理错误,可以使用函数pcall(protected call)来包装需要执行的代码。
  • pcall接收一个函数和要传递给后者的参数,并执行,执行结果:有错误、无错误;返回值true或者或false, errorinfo。

语法格式如下:

  1. if pcall(function_name, ….) then
  2. -- 没有错误
  3. else
  4. -- 一些错误
  5. end

简单实例:

  1. pcall(function(i) print(i) end, 33)
  2. 33
  3. true
  4. pcall(function(i) print(i) error('error..') end, 33)
  5. 33
  6. false stdin:1: error..

  1. function f()
  2. return false,2
  3. end
  4. if f()
  5. then print '1' else print '0'
  6. end

  • pcall以一种"保护模式"来调用第一个参数,因此pcall可以捕获函数执行中的任何错误。
  • 通常在错误发生时,希望落得更多的调试信息,而不只是发生错误的位置。但pcall返回时,它已经销毁了调用桟的部分内容。

🌴3.xpcall函数

Lua提供了xpcall函数,xpcall接收第二个参数——一个错误处理函数,当错误发生时,Lua会在调用桟展开(unwind)前调用错误处理函数,于是就可以在这个函数中使用debug库来获取关于错误的额外信息了。

简单实例1:

  1. xpcall(function(i) print(i) error('error..') end, function() print(debug.traceback()) end, 33)

简单实例2:

  1. function myfunction ()
  2. n = n/nil
  3. end
  4. function myerrorhandler( err )
  5. print( "ERROR:", err )
  6. end
  7. status = xpcall( myfunction, myerrorhandler )
  8. print( status)

🌵4.debug函数

  • debug库提供了两个通用的错误处理函数:
  • debug.debug:提供一个Lua提示符,让用户来检查错误的原因。
  • debug.traceback:根据调用桟来构建一个扩展的错误消息。

💬🌲🌳🌴🌵总结

以上就是今天要讲的内容,本文介绍了LuaLua错误处理,而Lua错误处理相关操作提供了程序的的错误提示能实现能使我们快速便捷地了解数据的函数和方法,我们务必掌握。另外如果上述有任何问题,请懂哥指教,不过没关系,主要是自己能坚持,更希望有一起学习的同学可以帮我指正,但是如果可以请温柔一点跟我讲,爱与和平是永远的主题,爱各位了。

相关文章