ការបង្រៀននេះពន្យល់អំពីការដោះស្រាយករណីលើកលែងនៅក្នុង Python ដោយប្រើប្លុក Try Except ដោយមានជំនួយពីឧទាហរណ៍នៃការសរសេរកម្មវិធី៖
ប្រភេទកំហុសពីរអាចបណ្តាលឱ្យកម្មវិធី Python បញ្ឈប់ភ្លាមៗ ពោលគឺ វាក្យសម្ព័ន្ធ កំហុស និង ករណីលើកលែង ។ នៅក្នុងមេរៀននេះ យើងនឹងពិភាក្សាអំពីប្រភេទកំហុសទីពីរ (Exceptions) ក្រោមប្រធានបទសំខាន់ៗមួយចំនួន។
យើងនឹងទទួលបានអត្ថប្រយោជន៍ជាច្រើនពីការដោះស្រាយករណីលើកលែងនៅក្នុងកម្មវិធីរបស់យើងដូចជា៖
- បង្កើតកម្មវិធីដ៏រឹងមាំ។
- បង្កើតកូដស្អាត និងគ្មានកំហុស។
Python Try Except
ដំណឹងល្អមួយគឺថា Python មានករណីលើកលែងជាច្រើនដែលភ្ជាប់មកជាមួយដើម្បីចាប់កំហុសនៅក្នុងកូដរបស់យើង។ ដូចគ្នានេះផងដែរ វាផ្តល់ឱ្យយើងនូវឱកាសដើម្បីបង្កើតការលើកលែងផ្ទាល់ខ្លួន នៅពេលដែលគ្មានការលើកលែងដែលភ្ជាប់មកជាមួយសមស្របនឹងតម្រូវការរបស់យើង។
តើអ្វីទៅជាករណីលើកលែង
ដូច្នេះតើអ្វីជាករណីលើកលែងនៅក្នុង Python? និយាយឱ្យចំទៅ រាល់ពេលដែលអ្នកបកប្រែ Python ព្យាយាមប្រតិបត្តិកូដមិនត្រឹមត្រូវ វាលើកករណីលើកលែង ហើយក្នុងករណីដែលករណីលើកលែងបែបនេះមិនត្រូវបានដោះស្រាយ វារំខានដល់លំហូរធម្មតានៃការណែនាំរបស់កម្មវិធី និងបោះពុម្ពការតាមដាន។
តោះបង្កើតកូដមិនត្រឹមត្រូវ ហើយមើលពីរបៀបដែលអ្នកបកប្រែ Python នឹងឆ្លើយតប។
បើកសែល Python ហើយដំណើរការកូដខាងក្រោម។
>>> 50/0
នេះគឺជាមួយក្នុងចំណោម កំហុសទូទៅបំផុតក្នុងការសរសេរកម្មវិធី។ លេខកូដខាងលើព្យាយាមបែងចែកលេខ 50 ដោយ 0 (សូន្យ)។ ពស់ថ្លាន់អថេរ openFile មុនពេលដែលវាត្រូវបានចាត់តាំង។
ល្បិចតូចមួយនៅទីនេះគឺប្រើកម្មវិធីដោះស្រាយករណីលើកលែងនៅខាងក្នុងប្លុកចុងក្រោយ។
def readFile(file_path): try: openFile = open(file_path,'r') # Open a file as read-only print(openFile.readline()) # Read first line of file content except FileNotFoundError as ex: print(ex) finally: try: print("Cleaning...") openFile.close() except: # catches all exceptions pass # Ignore this error because we don't care. if __name__ == '__main__': filePath = './text.txt' readFile(filePath)
ប្រសិនបើការសាកល្បងរបស់យើងបង្កើន FileNotFoundError នោះយើងនឹងមានលទ្ធផលដូចខាងក្រោម
លើកករណីលើកលែង
ដំណឹងល្អមួយអំពីការលើកលែង Python គឺថាយើងអាចមានចេតនា លើកពួកគេ។ ករណីលើកលែងត្រូវបានលើកឡើងជាមួយនឹង លើកឡើង ។
សេចក្តីថ្លែងការលើកឡើងមានវាក្យសម្ព័ន្ធដូចខាងក្រោម៖
raise [ExceptionName[(*args: Object)]]
បើកស្ថានីយ និងលើកវត្ថុករណីលើកលែងណាមួយពី ករណីលើកលែងដែលភ្ជាប់មកជាមួយ Python ។ ឧទាហរណ៍ ប្រសិនបើយើងលើក ZeroDivisionError៖
>>> raise ZeroDivisionError("Can't divide by zero")
យើងនឹងទទួលបាននូវការតាមដាន៖
ដូច្នេះ។ ហេតុអ្វីបានជាវាសំខាន់ក្នុងការលើកករណីលើកលែង?
- នៅពេលធ្វើការជាមួយការលើកលែងផ្ទាល់ខ្លួន។
- ក្នុងអំឡុងពេលត្រួតពិនិត្យអនាម័យ។
ថ្នាក់ការលើកលែងផ្ទាល់ខ្លួន
ករណីលើកលែងផ្ទាល់ខ្លួនគឺជារឿងមួយដែលអ្នកបង្កើតដើម្បីដោះស្រាយកំហុសដែលជាក់លាក់ចំពោះតម្រូវការរបស់អ្នក។ ល្បិចគឺយើងកំណត់ថ្នាក់ដែលកើតចេញពីវត្ថុ Exception បន្ទាប់មកយើងប្រើសេចក្តីថ្លែងការលើកឡើងដើម្បីលើកថ្នាក់លើកលែងរបស់យើង។
ឧបមាថាយើងចង់ពិនិត្យមើលការបញ្ចូលរបស់អ្នកប្រើ ហើយត្រូវប្រាកដថា តម្លៃបញ្ចូលគឺមិនអវិជ្ជមានទេ (ការត្រួតពិនិត្យអនាម័យ) ។ ជាការពិតណាស់ យើងអាចលើកករណីលើកលែង Python ValueError ប៉ុន្តែយើងនឹងចង់ប្ដូរតាមបំណងនូវកំហុសដោយផ្តល់ឱ្យវានូវឈ្មោះជាក់លាក់ និងពន្យល់ដោយខ្លួនឯងដូចជា InputIsNegativeError ។ ប៉ុន្តែករណីលើកលែងនេះមិនមែនជា Python ដែលភ្ជាប់មកជាមួយទេ។Exception។
ដូច្នេះដំបូង យើងបង្កើតថ្នាក់មូលដ្ឋានរបស់យើង ដែលនឹងចេញមកពី Exception។
class CustomError(Exception): "Base class exception for all exceptions of this module" pass
បន្ទាប់មកយើងបង្កើតថ្នាក់ exception របស់យើងដែលនឹងទទួលមរតកនៃថ្នាក់មូលដ្ឋាន ហើយនឹងដោះស្រាយកំហុសជាក់លាក់របស់យើង។
class InputIsNegativeError(CustomError): """Raised when User enters a negative value""" pass
តោះសាកល្បងនេះ
try: value = int(input()) if value 0: raise InputIsNegativeError # Raise exception if value is negative except InputIsNegativeError: # catch and handle exception print("Input value shouldn't be negative")
សំណើកូដខាងលើសម្រាប់ការបញ្ចូលរបស់អ្នកប្រើប្រាស់ ហើយពិនិត្យមើលថាតើវាអវិជ្ជមានឬអត់។ ប្រសិនបើជាការពិត វាបង្កើនការលើកលែងផ្ទាល់ខ្លួនរបស់យើង InputIsNegativeError ដែលក្រោយមកត្រូវបានចាប់បាននៅក្នុងសេចក្តីថ្លែងការលើកលែង។
ខាងក្រោមគឺជាកូដពេញលេញ៖
class CustomError(Exception): "Base class exception for all exceptions of this module" pass class InputIsNegativeError(CustomError): """Raised when User enters a negative value""" pass if __name__ == '__main__': try: value = int(input("Input a number: ")) if value 0: raise InputIsNegativeError # Raise exception if value is negative except InputIsNegativeError: # catch and handle exception print("Input value shouldn't be negative")
ប្រសិនបើតម្លៃបញ្ចូល គឺជាលេខអវិជ្ជមានដូចជា -1 បន្ទាប់មកយើងនឹងមានលទ្ធផល៖
ពិនិត្យមើលឯកសារ Python សម្រាប់ព័ត៌មានលម្អិតបន្ថែមអំពីករណីលើកលែងផ្ទាល់ខ្លួនរបស់ Python ។
សំណួរដែលគេសួរញឹកញាប់
សំណួរ #1) តើ Python ដោះស្រាយការលើកលែងដោយរបៀបណា?
ចម្លើយ៖ Python ដោះស្រាយការលើកលែងដោយប្រើ សាកល្បង-លើកលែងតែសេចក្តីថ្លែងការណ៍ ។ កូដដែលអាចលើកករណីលើកលែងត្រូវបានដាក់ និងប្រតិបត្តិក្នុង សាកល្បងប្លុក ខណៈពេលដែល ការលើកលែងប្លុក កាន់កូដដែលនឹងដោះស្រាយការលើកលែងប្រសិនបើមានកើតឡើង។
សំណួរ #2) តើអ្វីជាការលើកករណីលើកលែងនៅក្នុង Python? ដើម្បីប្រាប់យើងថាមានអ្វីមួយដែលមិននឹកស្មានដល់បានកើតឡើង។ យើងក៏អាចលើកករណីលើកលែងដោយចេតនាផងដែរ ដោយប្រើ លើកឡើង ។
សំណួរ #3) តើ Python ដោះស្រាយការលើកលែងច្រើនយ៉ាងដូចម្តេច?
ចម្លើយ៖ Python ដោះស្រាយការលើកលែងជាច្រើន។ដោយប្រើប្លុកតែមួយ លើកលែងតែប្លុក ឬច្រើនលើកលែងតែប្លុក។
សម្រាប់ប្លុកតែមួយ ការលើកលែងត្រូវបានឆ្លងកាត់ជា tuple៖ លើកលែងតែ (Exception1, Exception2,..,ExceptionN) និងការត្រួតពិនិត្យ Python សម្រាប់ការប្រកួតពីស្តាំទៅឆ្វេង។ ក្នុងករណីនេះ សកម្មភាពដូចគ្នានេះត្រូវបានធ្វើឡើងសម្រាប់ករណីលើកលែងនីមួយៗ។
វិធីមួយផ្សេងទៀតដើម្បីចាប់យកករណីលើកលែងទាំងអស់គឺត្រូវទុកឈ្មោះករណីលើកលែងបន្ទាប់ពីពាក្យគន្លឹះលើកលែង។
except: # handle all exceptions here
វិធីទីពីរគឺ ដើម្បីប្រើប្លុកលើកលែងសម្រាប់ករណីលើកលែងនីមួយៗ៖
except Exception1: # code to handle Exception1 goes here except Exception2: # code to handle Exception2 goes here except ExceptionN: # code to handle ExceptionN goes here
តាមវិធីនេះ អ្នកអាចធ្វើសកម្មភាពដាច់ដោយឡែកសម្រាប់ Exception នីមួយៗ។
សំណួរ #4) ហេតុអ្វីបានជាការដោះស្រាយ Exception មានសារៈសំខាន់នៅក្នុង Python?
ចម្លើយ៖ អត្ថប្រយោជន៍នៃការដោះស្រាយករណីលើកលែងនៅក្នុង Python គឺថាយើងអាចបង្កើតកម្មវិធីរឹងមាំ ស្អាត និងគ្មានកំហុស។ យើងនឹងមិនចង់ឱ្យកូដផលិតកម្មរបស់យើងគាំងដោយសារតែមានកំហុសមួយចំនួន ដូច្នេះយើងដោះស្រាយកំហុស ហើយបន្តដំណើរការកម្មវិធីរបស់យើង។
សំណួរ #5) តើអ្នកមិនអើពើនឹងករណីលើកលែងក្នុង Python ដោយរបៀបណា?
ចម្លើយ៖ ដើម្បីមិនអើពើករណីលើកលែងនៅក្នុង Python សូមប្រើពាក្យគន្លឹះ pass នៅក្នុងប្លុកលើកលែងតែ។ ឧបមាថាយើងចង់មិនអើពើនឹងការលើកលែង ValueError ។ យើងនឹងធ្វើវាតាមវិធីនេះ៖
except ValueError: pass
លុះត្រាតែអ្នកដឹងពីអ្វីដែលអ្នកកំពុងធ្វើ វាជាការអនុវត្តមិនល្អក្នុងការមិនអើពើនឹងករណីលើកលែង។ យ៉ាងហោចណាស់ សូមជូនដំណឹងដល់អ្នកប្រើប្រាស់អំពីកំហុសដែលអាចកើតមានទាំងអស់។
សេចក្តីសន្និដ្ឋាន
នៅក្នុងមេរៀននេះ យើងបានគ្របដណ្តប់៖ ការលើកលែង Python, Traceback; របៀបដោះស្រាយការលើកលែងជាមួយ សាកល្បង / លើកលែងតែ / Else / ចុងក្រោយ ប្លុក របៀប បង្កើន ករណីលើកលែង និងចុងក្រោយរបៀបបង្កើតការលើកលែងផ្ទាល់ខ្លួនរបស់យើង។
សូមអរគុណសម្រាប់ការអាន!
អ្នកបកប្រែយល់ឃើញថានេះជាប្រតិបត្តិការមិនត្រឹមត្រូវ ហើយលើកឡើង ZeroDivisionErrorរំខានកម្មវិធី និងបោះពុម្ពការតាមដាន។
យើងអាចមើលឃើញយ៉ាងច្បាស់ថា ZeroDivisionError គឺជាករណីលើកលែងដែលត្រូវបានលើកឡើង។ វាពិតជាវិធីផ្ទាល់ខ្លួនរបស់ Python ដើម្បីប្រាប់យើងថាវាមិនត្រជាក់ទេក្នុងការបែងចែកលេខដោយសូន្យ។ ទោះបីជានៅក្នុងភាសាផ្សេងទៀតដូចជា JavaScript ក៏ដោយ នេះមិនមែនជាកំហុសទេ។ ហើយ python ហាមប្រាមការអនុវត្តន៍នេះយ៉ាងតឹងរ៉ឹង។
វាសំខាន់ផងដែរក្នុងការដឹងថានេះគ្រាន់តែជាវត្ថុលើកលែង ហើយ Python មានវត្ថុបែបនេះជាច្រើននៅក្នុងខ្លួន។ សូមពិនិត្យមើលឯកសារផ្លូវការរបស់ Python នេះ ដើម្បីមើលករណីលើកលែងដែលភ្ជាប់មកជាមួយ Python ទាំងអស់។
ការយល់ដឹងពីការតាមដាន
មុនពេលដែលយើងឈានដល់ការដោះស្រាយករណីលើកលែង ខ្ញុំគិតថាវានឹងជួយឱ្យយល់ពីអ្វីដែលនឹងកើតឡើង ប្រសិនបើករណីលើកលែង មិនត្រូវបានដោះស្រាយ និងរបៀបដែល Python ប្រឹងប្រែងអស់ពីសមត្ថភាពដើម្បីជូនដំណឹងដល់យើងអំពីកំហុសរបស់យើង។
នៅពេលណាដែល Python ជួបប្រទះកំហុស វាបង្កើនករណីលើកលែងមួយ។ ប្រសិនបើករណីលើកលែងនេះមិនត្រូវបានដោះស្រាយទេ នោះវានឹងបង្កើតព័ត៌មានមួយចំនួនដែលហៅថា Traceback។ ដូច្នេះ តើការតាមដាននេះមានព័ត៌មានអ្វីខ្លះ?
វាមាន៖
- សារកំហុសដែលប្រាប់យើងពីអ្វីដែលករណីលើកលែងត្រូវបានលើកឡើង និងអ្វីដែលបានកើតឡើងមុនពេលករណីលើកលែងនេះគឺ បានលើកឡើង។
- លេខបន្ទាត់ផ្សេងៗនៃកូដដែលបណ្តាលឱ្យមានកំហុសនេះ។ កំហុសអាចបណ្តាលមកពីលំដាប់នៃការហៅមុខងារហៅថា call stack ដែលយើងនឹងពិភាក្សានៅពេលក្រោយនៅទីនេះ។
ទោះបីជាវាជាមានការភាន់ច្រលំបន្តិច យើងសន្យាថាឧទាហរណ៍បន្ទាប់នឹងនាំមកនូវពន្លឺបន្ថែមទៀតដល់ការយល់ដឹងរបស់យើង។
រំលឹកឡើងវិញនូវដានដែលបានបោះពុម្ពពីការបែងចែក 50 គុណនឹង 0 ខាងលើ យើងអាចឃើញថា ដានខាងក្រោយមានព័ត៌មានដូចខាងក្រោម៖
- File “”: វាប្រាប់យើងថាលេខកូដនេះត្រូវបានដំណើរការពីកុងសូលស្ថានីយ។
- បន្ទាត់ទី 1៖ វាប្រាប់យើងថាកំហុសបានកើតឡើងនៅក្នុងលេខបន្ទាត់នេះ។
- ZeroDivisionError៖ ការបែងចែកដោយ សូន្យ៖ វាប្រាប់យើងពីអ្វីដែលករណីលើកលែងត្រូវបានលើកឡើង និងអ្វីដែលបណ្តាលឱ្យវា។
តោះសាកល្បងឧទាហរណ៍មួយផ្សេងទៀត និង ប្រហែលជាមើលពីរបៀបដែល call stack មើលទៅ។ បើកកម្មវិធីនិពន្ធ បញ្ចូលកូដខាងក្រោម ហើយរក្សាទុកជា tracebackExp .py
def stack1(numb): # 1 div = 0 # 2 stack2(numb, div) # 3 def stack2(numb, div): # 5 compute = numb/div # 6 print(compute) # 7 if __name__ == '__main__': # 9 numb = 5 # 10 stack1(numb) # 11
បើកស្ថានីយក្នុងថតដែលឯកសារនេះត្រូវបានរកឃើញ ហើយដំណើរការ។
python tracebackExp.py
អ្នកនឹងឃើញដានខាងក្រោម៖
ការតាមដានខាងលើអាចមើលទៅហាក់ដូចជាច្របូកច្របល់ ប៉ុន្តែតាមពិតវាមិនមែនទេ។ Pythonistas បានបង្កើតនូវវិធីល្អបំផុតដើម្បីអាន traceback ដែលមកពី ខាងក្រោមឡើង ។ ដូច្នេះ ចូរយើងប្រើប្រាជ្ញានេះ ដើម្បីសាកល្បង និងយល់ពីអ្វីដែលដាននេះមានផ្តល់ជូន។
- នៅខាងក្រោមបំផុត យើងទទួលបានករណីលើកលែងដែលត្រូវបានលើកឡើង និងមូលហេតុដែលវាត្រូវបានលើកឡើង។
- ការផ្លាស់ទីឡើងលើ យើងទទួលបានឈ្មោះឯកសារ tracebackExp .py ដែលកំហុសនេះបានកើតឡើង ការគណនាដែលបណ្តាលឱ្យមានកំហុសនេះ compute = numb/div មុខងារ stack2 និងតំណភ្ជាប់លេខ 6 ដែលការគណនានេះត្រូវបានអនុវត្ត .
- ផ្លាស់ទីឡើងលើ យើងឃើញថាមុខងារ stack2 របស់យើង។ត្រូវបានហៅក្នុងអនុគមន៍ stack1 ក្នុងជួរលេខ 3។
- ផ្លាស់ទីទៅកំពូល យើងឃើញថាមុខងារ stack1 ត្រូវបានហៅក្នុងជួរលេខ 11។ module > ប្រាប់យើងថាវាជាឯកសារដែលកំពុងដំណើរការ។
Common Python Exceptions
បណ្ណាល័យ Python កំណត់ករណីលើកលែងដែលមានស្រាប់យ៉ាងច្រើន។ អ្នកអាចពិនិត្យមើលឯកសារ Python ឬហៅមុខងារ local () ដែលភ្ជាប់មកជាមួយដូចខាងក្រោម៖
>>> dir(locals()['__builtins__'])
យើងនឹងមិនព្យាយាមដោះស្រាយការលើកលែងទាំងអស់នេះទេ ប៉ុន្តែយើងនឹងឃើញករណីលើកលែងធម្មតាមួយចំនួន ដែលអ្នកទំនងជានឹងជួប។
#1) TypeError
វាត្រូវបានលើកឡើងនៅពេលដែលប្រតិបត្តិការ ឬមុខងារត្រូវបានអនុវត្តចំពោះវត្ថុនៃប្រភេទមិនសមរម្យ។
ឧទាហរណ៍ 1
ពិចារណាកម្មវិធីខាងក្រោម។ វាត្រូវការភាគលាភ និងផ្នែកចែក បន្ទាប់មកគណនា និងបោះពុម្ពលទ្ធផលនៃការបែងចែកភាគលាភដោយអ្នកចែក។
def compute_division(): dividend = int(input("Enter the dividend: ")) # cast string to int divisor = input("Enter the divisor: ") # no casting # Compute division result = dividend/divisor # print result print("The result of {}/{} is: {}".format(dividend, divisor, result)) if __name__ == '__main__': result = compute_division()
យើងស្នើសុំតម្លៃនៃភាគលាភ និងផ្នែកចែកពីអ្នកប្រើប្រាស់ ប៉ុន្តែយើងភ្លេចដាក់ខ្សែរចែក តម្លៃទៅជាចំនួនគត់។ ដូច្នេះ យើងបញ្ចប់ដោយប្រភេទនៃភាគលាភគឺចំនួនគត់( int ) និងប្រភេទអ្នកចែកជាខ្សែអក្សរ( str )។ បន្ទាប់មកយើងទទួលបាន TypeError ព្រោះថា division operator (/) មិនដំណើរការលើ strings។
វាប្រហែលជាចាប់អារម្មណ៍ដែលអ្នកដឹងថាមិនដូច Python ទេ Javascript មាន Type Coercion ដែលជាមូលដ្ឋានបំប្លែងប្រភេទនៃ operand ទៅជាតម្លៃសមមូលនៃប្រភេទ operand ផ្សេងទៀត នៅពេលដែល operands មាន។ប្រភេទផ្សេងគ្នា។
#2) ValueError
វាលើកឡើងនៅពេលដែលប្រតិបត្តិការ ឬមុខងារទទួលបានអាគុយម៉ង់ដែលមានប្រភេទត្រឹមត្រូវ ប៉ុន្តែតម្លៃមិនសមរម្យ។
ឧទាហរណ៍ 2
ពិចារណាកម្មវិធីរបស់យើងក្នុង ឧទាហរណ៍ 1 ខាងលើ។
ប្រសិនបើអ្នកប្រើបញ្ចូលតម្លៃអក្សរក្រមលេខសម្រាប់ភាគលាភដូចជា '3a' នោះកម្មវិធីរបស់យើងនឹងកើនឡើង ករណីលើកលែង ValueError ។ នេះគឺដោយសារតែ ទោះបីជាវិធីសាស្ត្រ Python int() យកលេខ ឬខ្សែអក្សរណាមួយ ហើយត្រឡប់វត្ថុចំនួនគត់ក៏ដោយ តម្លៃខ្សែអក្សរមិនគួរមានអក្សរ ឬតម្លៃដែលមិនមែនជាលេខទេ។
#3) AttributeError
ករណីលើកលែងនេះត្រូវបានលើកឡើង ខណៈពេលដែលផ្តល់ ឬយោងគុណលក្ខណៈដែលមិនមាន។
ឧទាហរណ៍ 3
ពិចារណាកម្មវិធី ខាងក្រោម។ វាយកជាលេខ ហើយគណនាឫសការ៉េរបស់វាដោយប្រើម៉ូឌុលគណិតវិទ្យា Python
import math # import math library to gain access to its code def compute_square_root(number): # compute the square root using the math library result = math.sqr(number) return result if __name__ == '__main__': # get input to compute from user number = int(input("Compute Square root of: ")) # call function to compute square root
នៅពេលអ្នកប្រើប្រាស់បញ្ចូលលេខ កម្មវិធីរបស់យើងព្យាយាមប្រើមុខងារពីម៉ូឌុលគណិតវិទ្យាដើម្បីគណនាឫសការ៉េរបស់វា ប៉ុន្តែគ្រាន់តែនៅទីនេះ។ យើងបានធ្វើកំហុស។ ជំនួសឱ្យ sqrt យើងបានវាយបញ្ចូលកំហុស sqr ដែលមិនមាននៅក្នុងម៉ូឌុលគណិតវិទ្យា។
ដូច្នេះ យើងកំពុងព្យាយាមយោងគុណលក្ខណៈ sqr ដែលមិនមាន និងដឹកនាំ ចំពោះករណីលើកលែង AttributeError ត្រូវបានលើកឡើង។ ភាគច្រើននៃពួកយើងមានកំហុសបែបនេះច្រើន។ ដូច្នេះ អ្នកមិននៅម្នាក់ឯងទេ។
ដោះស្រាយករណីលើកលែងដោយសាកល្បង លើកលែងតែ
ក្នុងនាមជាអ្នកសរសេរកម្មវិធី រឿងមួយដែលពួកយើងភាគច្រើននឹងចំណាយពេលវេលារបស់យើងគឺការសរសេរកូដដ៏រឹងមាំដែលជាធន់។ កូដដែលមិនខូចដោយសារកំហុសមួយចំនួន។ នៅក្នុង Python យើងអាចសម្រេចបានដោយការបញ្ចូលសេចក្តីថ្លែងការណ៍របស់យើងនៅក្នុង try – លើកលែងតែ statement។
Python Try-Except statement
សេចក្តីថ្លែងការណ៍ try-except មានរចនាសម្ព័ន្ធដូចខាងក្រោម៖
try: #your code goes here except """Specify exception type(s) here""": #handle exception here
តោះបញ្ចូលកូដនៅក្នុង tracebackExp .py នៅខាងក្នុង try-exception statement។
def stack1(numb): # 1 div = 0 # 2 stack2(numb, div) # 3 def stack2(numb, div): # 5 try: # 6 compute = numb/div # 7 print(compute) # 8 except ZeroDivisionError as zde: # 9 print(zde) # 10 if __name__ == '__main__': # 12 numb = 5 # 13 stack1(numb) # 14 print("program continuous") # 15
ការដំណើរការកូដនេះនឹងបង្កើតលទ្ធផល
នេះជារបៀបដែល try-exception statement ដំណើរការ។ Python ប្រតិបត្តិកូដនៅក្នុង try block line 7-8 ។ ប្រសិនបើរកមិនឃើញលេខកូដមិនត្រឹមត្រូវទេ នោះលេខកូដនៅក្នុងប្លុកលើកលែងតែ បន្ទាត់ 10 ត្រូវបានរំលង ហើយការប្រតិបត្តិនៅតែបន្ត។
ប៉ុន្តែប្រសិនបើលេខកូដមិនត្រឹមត្រូវត្រូវបានរកឃើញ នោះការប្រតិបត្តិភ្លាមៗនឹងឈប់នៅក្នុង សាកល្បងទប់ស្កាត់ ហើយពិនិត្យមើលថាតើករណីលើកលែងដែលបានលើកឡើងត្រូវគ្នាជាមួយនឹងអ្វីដែលយើងបានផ្តល់នៅក្នុងសេចក្តីថ្លែងការលើកលែង បន្ទាត់ 9 ។ ប្រសិនបើវាត្រូវគ្នា នោះប្លុកលើកលែងតែត្រូវបានប្រតិបត្តិ ហើយបន្ត។ ប្រសិនបើវាមិនដំណើរការទេ នោះកម្មវិធីនឹងរំខាន។
ការសាកល្បងប្លុកជាធម្មតាមានលេខកូដដែលអាចបង្កើនករណីលើកលែងមួយ ខណៈពេលដែលប្លុកលើកលែងចាប់ និងដោះស្រាយការលើកលែង។
ការគ្រប់គ្រងច្រើន ករណីលើកលែង លើកលែងតែ
យើងអាចដោះស្រាយករណីលើកលែងជាច្រើនដោយប្រើ "លើកលែងតែ" តែមួយ ឬ "លើកលែងតែ" ច្រើន។ វាទាំងអស់គឺអាស្រ័យលើរបៀបដែលអ្នកចង់ដោះស្រាយការលើកលែងនីមួយៗ។
#1) ដោះស្រាយការលើកលែងច្រើនដោយលើកលែងតែមួយ
try: #your code goes here except(Exception1[, Exception2[,...ExceptionN]]]): #handle exception here
វិធីសាស្ត្រនេះត្រូវបានប្រើនៅពេលដែលយើងសង្ស័យថាកូដរបស់យើងអាចលើកករណីលើកលែងផ្សេងៗគ្នា ហើយយើងចង់ធ្វើសកម្មភាពដូចគ្នានៅក្នុងករណីនីមួយៗ។ ដូច្នេះ ប្រសិនបើអ្នកបកប្រែ Python រកឃើញការផ្គូផ្គង នោះកូដដែលសរសេរក្នុងប្លុកលើកលែងតែនឹងដំណើរការ។
សូមពិចារណាឧទាហរណ៍កូដ Python ខាងក្រោម
def get_fraction(value, idx): arr = [4,5,2,0] # a list of numbers idx_value = arr[idx] # if idx is > arr length, IndexError will be raised value/idx_value # if idx_value == 0, ZeroDivisionError will be raised if __name__ =='__main__': # set 'value' and 'idx' value = 54 idx = 3 # call function in a try-except statement. try: result = get_fraction(value, idx) print("Fraction is ", result) except (IndexError, ZeroDivisionError) as ex: print(ex)
យើងមានពីរ ករណីលើកលែងដែលអាចកើតមានដែលអាចត្រូវបានលើកឡើងនៅទីនេះ ZeroDivisionError និង IndexError ។ ប្រសិនបើករណីលើកលែងទាំងនេះត្រូវបានលើកឡើង នោះប្លុកលើកលែងតែនឹងត្រូវបានប្រតិបត្តិ។
នៅក្នុងកូដខាងលើ idx=3 ដូច្នេះ idx_ តម្លៃ ក្លាយជា 0 និង តម្លៃ /idx_ value នឹងបង្កើន ZeroDivisionError
#2) ការដោះស្រាយការលើកលែងច្រើនដោយមានករណីលើកលែងច្រើន
try: #your code goes here except Exception1: #handle exception1 here except Exception2: #handle exception2 here except ExceptionN: #handle exceptionN here
ប្រសិនបើយើងចង់ដោះស្រាយ ករណីលើកលែងនីមួយៗដាច់ដោយឡែកពីគ្នា នោះជារបៀបដែលអ្នកអាចធ្វើបាន។
សូមពិចារណាឧទាហរណ៍កូដ Python ខាងក្រោម
def get_fraction(value, idx): arr = [4,5,2,0] # a list of numbers idx_value = arr[idx] # if idx is > arr length, IndexError will be raised value/idx_value # if idx_value == 0, ZeroDivisionError will be raised if __name__ =='__main__': # set 'value' and 'idx' value = 54 idx = 5 # call function in a try-excepts statement. try: result = get_fraction(value, idx) print("Fraction is ", result) except IndexError: print("idx of {} is out of range".format(idx)) except ZeroDivisionError: print("arr[{}] is 0. Hence, can't divide by zero".format(idx)) except Exception as ex: print(ex) print("Not sure what happened so not safe to continue, \ app will be interrupted") raise ex
យើងកត់សំគាល់នៅទីនេះថា Exception ត្រូវបានប្រើនៅក្នុងពាក្យលើកលែងចុងក្រោយ . នេះគឺដោយសារតែវត្ថុលើកលែង Exception ត្រូវនឹងករណីលើកលែងណាមួយ។ ដោយហេតុផលនេះ វាគួរតែចុងក្រោយជានិច្ច ចាប់តាំងពី Python នឹងបញ្ឈប់ការពិនិត្យមើលកម្មវិធីដោះស្រាយករណីលើកលែងផ្សេងទៀតនៅពេលមួយត្រូវគ្នា។
នៅក្នុងកូដខាងលើ idx=5 ដូច្នេះ arr[idx ] នឹងលើកឡើង IndexError ដោយសារតែ idx ធំជាងប្រវែងនៃបញ្ជី arr
ផងដែរ មិនប្រាកដថាអ្វីដែលករណីលើកលែងត្រូវបានលើកឡើងដោយកម្មវិធីរបស់អ្នកគឺមិនមានសុវត្ថិភាពក្នុងការបន្តការប្រតិបត្តិនោះទេ។ នោះហើយជាមូលហេតុដែលយើងមានប្រភេទ Exception ដើម្បីចាប់យកករណីលើកលែងដែលមិនអាចទាយទុកជាមុនបាន។ បន្ទាប់មកយើងជូនដំណឹងអ្នកប្រើប្រាស់ និងរំខានកម្មវិធីដោយបង្កើនករណីលើកលែងដូចគ្នា។
សាកល្បងសេចក្តីថ្លែងការផ្សេងទៀត
នេះគឺជា មុខងារស្រេចចិត្ត នៃការដោះស្រាយករណីលើកលែង និងអនុញ្ញាតឱ្យអ្នកបន្ថែមកូដដែលអ្នកចង់ ដំណើរការនៅពេលដែលគ្មានកំហុសកើតឡើង។ ប្រសិនបើមានកំហុសកើតឡើង ប្លុកផ្សេងទៀតនេះនឹងមិនដំណើរការទេ។
ពិចារណាឧទាហរណ៍កូដ Python ខាងក្រោម បើកកម្មវិធីនិពន្ធរបស់អ្នក ហើយរក្សាទុកកូដជា elseTry.py
def fraction_of_one(divisor): value = 1/divisor # if divisor is zero, ZeroDivisionError will be raised return value if __name__ == '__main__': while True: try: # Get input from the user. # if input is not a valid argument for int(), ValueError will be raised divisor = int(input("Enter a divisor: ")) # call our function to compute the fraction value = fraction_of_one(divisor) except (ValueError, ZeroDivisionError): print("Input can't be zero and should be a valid literal for int(). Please, try again!") else: print("Value: ", value) break
យើងទទួលបានការបញ្ចូលពីអ្នកប្រើប្រាស់ ហើយប្រើវាដើម្បីបែងចែក 1។ យើងមានករណីលើកលែងពីរដែលអាចកើតមាននៅទីនេះ ការបញ្ចូលរបស់អ្នកប្រើមិនត្រឹមត្រូវដែលនឹងបង្កឱ្យមាន ValueError និង សូន្យ(0) ដែលនឹងបណ្តាលឱ្យ កំហុសផ្នែកសូន្យ ។ សេចក្តីថ្លែងការណ៍លើកលែងរបស់យើងដោះស្រាយកំហុសទាំងនេះ។
ឥឡូវនេះ យើងចង់បោះពុម្ពតម្លៃនៃ តម្លៃ ។ ប្លុកផ្សេងទៀតរបស់យើងធ្វើឱ្យប្រាកដថាវាត្រូវបានបោះពុម្ពលុះត្រាតែការព្យាយាមប្លុករបស់យើងដំណើរការដោយគ្មានកំហុស។ នេះសំខាន់ព្រោះប្រសិនបើមានកំហុសកើតឡើងនៅក្នុងប្លុកសាកល្បងរបស់យើង តម្លៃ នឹងមិនត្រូវបានកំណត់។ ដូច្នេះ ការចូលប្រើវានឹងបង្កើនកំហុសមួយទៀត។
ដំណើរការកូដខាងលើដោយប្រើ Python elseTry.py
លទ្ធផលខាងលើបង្ហាញថា សម្រាប់ការបញ្ចូលដំបូង យើងវាយ 0 ហើយចុច ENTER។ ចាប់តាំងពីផ្នែករបស់យើងទទួលបាន 0, 1/divisor បានលើកឡើង zeroDivisionError ។ ការបញ្ចូលទីពីររបស់យើងគឺ k ដែលមិនត្រឹមត្រូវសម្រាប់ int () ដូច្នេះករណីលើកលែង ValueError ត្រូវបានលើកឡើង។
ប៉ុន្តែការបញ្ចូលចុងក្រោយរបស់យើងគឺ 9 ដែលត្រឹមត្រូវ និងជា លទ្ធផល យើងទទួលបានតម្លៃ “ តម្លៃ ” បោះពុម្ពជា 0.1111111111111111
ព្យាយាមចុងក្រោយសេចក្តីថ្លែងការណ៍
នេះក៏ជា មុខងារជាជម្រើស នៃការដោះស្រាយករណីលើកលែង ហើយនឹងដំណើរការជានិច្ច មិនថាមានអ្វីកើតឡើងនៅក្នុងកម្មវិធីដោះស្រាយករណីលើកលែងនោះទេ។
នោះគឺ៖
- ថាតើករណីលើកលែងកើតឡើងឬអត់
- ទោះបីជាមានការហៅ 'ត្រឡប់' នៅក្នុងប្លុកផ្សេងទៀតក៏ដោយ។
- ទោះបីជាស្គ្រីបត្រូវបានបិទនៅក្នុងប្លុកផ្សេងទៀត
ដូច្នេះ ប្រសិនបើយើងមានលេខកូដដែលយើងចង់ដំណើរការក្នុងគ្រប់ស្ថានភាពទាំងអស់ ទីបំផុតប្លុកគឺជាបុរសរបស់យើង។ ប្លុកនេះភាគច្រើនត្រូវបានប្រើសម្រាប់ការសម្អាតដូចជាការបិទឯកសារ។
សូមពិចារណាឧទាហរណ៍កូដ Python ខាងក្រោម
def readFile(file_path): try: openFile = open(file_path,'r') # Open a file as read-only print(openFile.readline()) # Read first line of file content except FileNotFoundError as ex: print(ex) finally: print("Cleaning...") openFile.close() if __name__ == '__main__': filePath = './text.txt' readFile(filePath)
កូដនេះព្យាយាមបើក និងអានឯកសារ text.txt នៅក្នុងថតបច្ចុប្បន្នរបស់វា។ ប្រសិនបើមានឯកសារនោះ កម្មវិធីរបស់យើងនឹងបោះពុម្ពជួរទីមួយនៃឯកសារ បន្ទាប់មកប្លុករបស់យើងនឹងដំណើរការ ហើយបិទឯកសារ។
និយាយថាយើងមានឯកសារមួយហៅថា text.txt នៅក្នុងថតដែលឯកសារកម្មវិធីនេះ គឺ និង មាន ហេឡូ។ ប្រសិនបើយើងដំណើរការកម្មវិធី នោះយើងនឹងមានលទ្ធផល
ឧទាហរណ៍នេះត្រូវបានជ្រើសរើសដោយចេតនា ព្រោះខ្ញុំចង់ឱ្យយើងដោះស្រាយបញ្ហាតូចមួយដែលអាចកើតឡើងនៅពេលបិទឯកសារចុងក្រោយ- block។
ប្រសិនបើឯកសារមិនមានទេ ការលើកលែង FileNotFoundError នឹងត្រូវបានលើកឡើង ហើយអថេរ openFile នឹងមិនត្រូវបានកំណត់ ហើយនឹងមិនមែនជាឯកសារទេ។ វត្ថុ។ ដូច្នេះ ការព្យាយាមបិទវានៅក្នុងប្លុកចុងក្រោយនឹងលើកករណីលើកលែង UnboundLocalError ដែលជាប្រភេទរងនៃ NameError ។
នេះជាមូលដ្ឋាននិយាយថាយើងកំពុងព្យាយាមយោង នេះ។