Lua - การจัดการข้อผิดพลาด

ต้องการการจัดการข้อผิดพลาด

การจัดการข้อผิดพลาดมีความสำคัญมากเนื่องจากการดำเนินการในโลกแห่งความเป็นจริงมักจะต้องใช้การดำเนินการที่ซับซ้อนซึ่งรวมถึงการดำเนินการไฟล์ธุรกรรมฐานข้อมูลและการเรียกใช้บริการเว็บ

ในการเขียนโปรแกรมใด ๆ มักจะมีข้อกำหนดสำหรับการจัดการข้อผิดพลาด ข้อผิดพลาดสามารถมีได้สองประเภท ได้แก่

  • ข้อผิดพลาดทางไวยากรณ์
  • ข้อผิดพลาดเวลาทำงาน

ข้อผิดพลาดทางไวยากรณ์

ข้อผิดพลาดทางไวยากรณ์เกิดขึ้นเนื่องจากการใช้ส่วนประกอบโปรแกรมต่างๆเช่นตัวดำเนินการและนิพจน์อย่างไม่เหมาะสม ตัวอย่างง่ายๆสำหรับข้อผิดพลาดทางไวยากรณ์แสดงอยู่ด้านล่าง

a == 2

ดังที่คุณทราบมีความแตกต่างระหว่างการใช้คำว่า "เท่ากับ" และ "เท่ากับ" สองเท่า การใช้อย่างใดอย่างหนึ่งแทนสิ่งอื่นอาจทำให้เกิดข้อผิดพลาด หนึ่ง "เท่ากับ" หมายถึงการกำหนดในขณะที่ "เท่ากับ" สองเท่าหมายถึงการเปรียบเทียบ ในทำนองเดียวกันเรามีนิพจน์และฟังก์ชันที่มีวิธีการใช้งานที่กำหนดไว้ล่วงหน้า

ตัวอย่างอื่นสำหรับข้อผิดพลาดทางไวยากรณ์แสดงไว้ด้านล่าง -

for a= 1,10
   print(a)
end

เมื่อเรารันโปรแกรมข้างต้นเราจะได้ผลลัพธ์ดังต่อไปนี้ -

lua: test2.lua:2: 'do' expected near 'print'

ข้อผิดพลาดของไวยากรณ์นั้นจัดการได้ง่ายกว่าข้อผิดพลาดขณะรันเนื่องจากล่าม Lua ค้นหาข้อผิดพลาดได้ชัดเจนกว่าในกรณีของข้อผิดพลาดรันไทม์ จากข้อผิดพลาดข้างต้นเราสามารถทราบได้อย่างง่ายดายว่าการเพิ่มคำสั่งdoก่อนพิมพ์คำสั่งเป็นสิ่งที่จำเป็นตามโครงสร้าง Lua

ข้อผิดพลาดเวลาทำงาน

ในกรณีที่เกิดข้อผิดพลาดรันไทม์โปรแกรมจะดำเนินการสำเร็จ แต่อาจทำให้เกิดข้อผิดพลาดรันไทม์เนื่องจากความผิดพลาดในการป้อนข้อมูลหรือฟังก์ชันที่จัดการไม่ถูกต้อง ตัวอย่างง่ายๆในการแสดงข้อผิดพลาดรันไทม์แสดงอยู่ด้านล่าง

function add(a,b)
   return a+b
end

add(10)

เมื่อเราสร้างโปรแกรมมันจะสร้างสำเร็จและรัน เมื่อรันจะแสดงข้อผิดพลาดขณะทำงาน

lua: test2.lua:2: attempt to perform arithmetic on local 'b' (a nil value)
stack traceback:
	test2.lua:2: in function 'add'
	test2.lua:5: in main chunk
	[C]: ?

นี่เป็นข้อผิดพลาดรันไทม์ซึ่งเกิดขึ้นเนื่องจากไม่ผ่านสองตัวแปร b คาดว่าพารามิเตอร์และที่นี่เป็นศูนย์และสร้างข้อผิดพลาด

ฟังก์ชัน Assert และ Error

เพื่อจัดการกับข้อผิดพลาดเรามักใช้สองฟังก์ชัน - assert และ error. ตัวอย่างง่ายๆแสดงไว้ด้านล่าง

local function add(a,b)
   assert(type(a) == "number", "a is not a number")
   assert(type(b) == "number", "b is not a number")
   return a+b
end

add(10)

เมื่อเรารันโปรแกรมข้างต้นเราจะได้ผลลัพธ์ข้อผิดพลาดดังต่อไปนี้

lua: test2.lua:3: b is not a number
stack traceback:
	[C]: in function 'assert'
	test2.lua:3: in function 'add'
	test2.lua:6: in main chunk
	[C]: ?

error (message [, level])ยุติฟังก์ชันป้องกันล่าสุดที่เรียกและส่งกลับข้อความเป็นข้อความแสดงข้อผิดพลาด ข้อผิดพลาดของฟังก์ชันนี้จะไม่ส่งกลับ โดยปกติข้อผิดพลาดจะเพิ่มข้อมูลบางอย่างเกี่ยวกับตำแหน่งข้อผิดพลาดที่จุดเริ่มต้นของข้อความ อาร์กิวเมนต์ระดับระบุวิธีรับตำแหน่งข้อผิดพลาด ด้วยระดับ 1 (ค่าเริ่มต้น) ตำแหน่งข้อผิดพลาดคือตำแหน่งที่เรียกใช้ฟังก์ชันข้อผิดพลาด ระดับ 2 ชี้ข้อผิดพลาดไปยังตำแหน่งที่เรียกใช้ฟังก์ชันที่เรียกว่าข้อผิดพลาด และอื่น ๆ การผ่านระดับ 0 จะหลีกเลี่ยงการเพิ่มข้อมูลตำแหน่งข้อผิดพลาดลงในข้อความ

pcall และ xpcall

ในการเขียนโปรแกรม Lua เพื่อหลีกเลี่ยงข้อผิดพลาดเหล่านี้และจัดการข้อผิดพลาดเราจำเป็นต้องใช้ฟังก์ชัน pcall หรือ xpcall

pcall (f, arg1, ...)ฟังก์ชันเรียกใช้ฟังก์ชันที่ร้องขอในโหมดป้องกัน หากข้อผิดพลาดบางอย่างเกิดขึ้นในฟังก์ชัน f จะไม่เกิดข้อผิดพลาด เพียงแค่ส่งกลับสถานะของข้อผิดพลาด ตัวอย่างง่ายๆโดยใช้ pcall แสดงไว้ด้านล่าง

function myfunction ()
   n = n/nil
end

if pcall(myfunction) then
   print("Success")
else
	print("Failure")
end

เมื่อเรารันโปรแกรมข้างต้นเราจะได้ผลลัพธ์ดังต่อไปนี้

Failure

xpcall (f, err)ฟังก์ชันเรียกใช้ฟังก์ชันที่ร้องขอและยังตั้งค่าตัวจัดการข้อผิดพลาด ข้อผิดพลาดใด ๆ ภายใน f จะไม่ถูกเผยแพร่ แทน xpcall จับข้อผิดพลาดเรียกฟังก์ชัน err ด้วยออบเจ็กต์ข้อผิดพลาดดั้งเดิมและส่งคืนรหัสสถานะ

ตัวอย่างง่ายๆสำหรับ xpcall แสดงไว้ด้านล่าง

function myfunction ()
   n = n/nil
end

function myerrorhandler( err )
   print( "ERROR:", err )
end

status = xpcall( myfunction, myerrorhandler )
print( status)

เมื่อเรารันโปรแกรมข้างต้นเราจะได้ผลลัพธ์ดังต่อไปนี้

ERROR:	test2.lua:2: attempt to perform arithmetic on global 'n' (a nil value)
false

ในฐานะโปรแกรมเมอร์สิ่งสำคัญที่สุดคือต้องแน่ใจว่าคุณดูแลจัดการข้อผิดพลาดที่เหมาะสมในโปรแกรมที่คุณเขียน การใช้การจัดการข้อผิดพลาดสามารถทำให้มั่นใจได้ว่าเงื่อนไขที่ไม่คาดคิดที่อยู่นอกเหนือเงื่อนไขขอบเขตจะได้รับการจัดการโดยไม่รบกวนผู้ใช้โปรแกรม