अमृत ​​- त्रुटियों को संभालना

अमृत ​​में तीन त्रुटि तंत्र हैं: त्रुटियां, फेंकता और बाहर निकलता है। आइए हम प्रत्येक तंत्र को विस्तार से देखें।

त्रुटि

जब कोड में असाधारण चीजें होती हैं तो त्रुटियों (या अपवादों) का उपयोग किया जाता है। एक स्ट्रिंग में एक संख्या जोड़ने की कोशिश करके एक नमूना त्रुटि प्राप्त की जा सकती है -

IO.puts(1 + "Hello")

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित त्रुटि पैदा करता है -

** (ArithmeticError) bad argument in arithmetic expression
   :erlang.+(1, "Hello")

यह एक इनबिल्ट त्रुटि का नमूना था।

उठी हुई त्रुटियाँ

हम कर सकते हैं raiseफ़ंक्शंस का उपयोग करते हुए त्रुटियाँ। आइए हम इसे समझने के लिए एक उदाहरण पर विचार करें -

#Runtime Error with just a message
raise "oops"  # ** (RuntimeError) oops

अन्य त्रुटियों को बढ़ाने / त्रुटि नाम और कीवर्ड तर्कों की एक सूची के साथ उठाया जा सकता है

#Other error type with a message
raise ArgumentError, message: "invalid argument foo"

आप अपनी त्रुटियों को भी परिभाषित कर सकते हैं और उन्हें बढ़ा सकते हैं। निम्नलिखित उदाहरण पर विचार करें -

defmodule MyError do
   defexception message: "default message"
end

raise MyError  # Raises error with default message
raise MyError, message: "custom message"  # Raises error with custom message

बचाव की त्रुटियाँ

हम नहीं चाहते हैं कि हमारे कार्यक्रम अचानक छोड़ दिए जाएं, बल्कि त्रुटियों को सावधानी से संभालने की जरूरत है। इसके लिए हम एरर हैंडलिंग का उपयोग करते हैं। हमrescue त्रुटियों का उपयोग कर try/rescueनिर्माण। आइए इसे समझने के लिए निम्नलिखित उदाहरण पर विचार करें -

err = try do
   raise "oops"
rescue
   e in RuntimeError -> e
end

IO.puts(err.message)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

oops

हमने पैटर्न मिलान का उपयोग करके बचाव वक्तव्य में त्रुटियों को संभाला है। यदि हमारे पास त्रुटि का कोई उपयोग नहीं है, और बस इसे पहचान के उद्देश्यों के लिए उपयोग करना चाहते हैं, तो हम फॉर्म का उपयोग भी कर सकते हैं -

err = try do
   1 + "Hello"
rescue
   RuntimeError -> "You've got a runtime error!"
   ArithmeticError -> "You've got a Argument error!"
end

IO.puts(err)

जब प्रोग्राम के ऊपर चल रहा है, यह निम्नलिखित परिणाम पैदा करता है -

You've got a Argument error!

NOTE- अमृत मानक पुस्तकालय में अधिकांश कार्य दो बार लागू किए जाते हैं, एक बार ट्यूपल्स को लौटाने और दूसरी बार त्रुटियों को बढ़ाते हुए। उदाहरण के लिए,File.read और यह File.read!कार्य करता है। यदि फ़ाइल सफलतापूर्वक पढ़ी गई थी और यदि कोई त्रुटि सामने आई थी, तो पहले ने एक टपल लौटाया था, इस टपल का उपयोग त्रुटि का कारण बताने के लिए किया गया था। यदि कोई त्रुटि सामने आई, तो दूसरे ने एक त्रुटि उठाई।

यदि हम पहले फ़ंक्शन दृष्टिकोण का उपयोग करते हैं, तो हमें त्रुटि से मेल खाने वाले पैटर्न के लिए मामले का उपयोग करने की आवश्यकता है और उसके अनुसार कार्रवाई करनी चाहिए। दूसरे मामले में, हम त्रुटि प्रवण कोड के लिए बचाव बचाव दृष्टिकोण का उपयोग करते हैं और तदनुसार त्रुटियों को संभालते हैं।

फेंकता

अमृत ​​में, एक मान फेंका जा सकता है और बाद में पकड़ा जा सकता है। थ्रो एंड कैच उन परिस्थितियों के लिए आरक्षित हैं जहां फेंक और कैच का उपयोग करके जब तक किसी मूल्य को प्राप्त करना संभव नहीं है।

पुस्तकालयों के साथ हस्तक्षेप करते हुए, उदाहरण अभ्यास में काफी असामान्य हैं। उदाहरण के लिए, अब मान लेते हैं कि Enum मॉड्यूल ने कोई मान प्राप्त करने के लिए कोई API प्रदान नहीं किया है और हमें संख्याओं की सूची में 13 के पहले एकाधिक को खोजने की आवश्यकता है -

val = try do
   Enum.each 20..100, fn(x) ->
      if rem(x, 13) == 0, do: throw(x)
   end
   "Got nothing"
catch
   x -> "Got #{x}"
end

IO.puts(val)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Got 26

बाहर जाएं

जब एक प्रक्रिया "प्राकृतिक कारणों" की मृत्यु हो जाती है (उदाहरण के लिए, बिना किसी अपवाद के), यह एक निकास संकेत भेजता है। एक प्रक्रिया भी स्पष्ट रूप से एक निकास संकेत भेजकर मर सकती है। आइए हम निम्नलिखित उदाहरण पर विचार करें -

spawn_link fn -> exit(1) end

ऊपर के उदाहरण में, लिंक की गई प्रक्रिया 1. के मूल्य के साथ एक निकास संकेत भेजकर मर गई। ध्यान दें कि बाहर निकलने की कोशिश / पकड़ का उपयोग करके "पकड़ा" भी जा सकता है। उदाहरण के लिए -

val = try do
   exit "I am exiting"
catch
   :exit, _ -> "not really"
end

IO.puts(val)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

not really

उपरांत

कभी-कभी यह सुनिश्चित करना आवश्यक है कि कुछ कार्रवाई के बाद एक संसाधन को साफ किया जाता है जो संभावित रूप से एक त्रुटि उठा सकता है। कोशिश / निर्माण के बाद आप ऐसा करने की अनुमति देता है। उदाहरण के लिए, हम एक फाइल खोल सकते हैं और एक क्लॉज के बाद इसे बंद कर सकते हैं-भले ही कुछ गलत हो।

{:ok, file} = File.open "sample", [:utf8, :write]
try do
   IO.write file, "olá"
   raise "oops, something went wrong"
after
   File.close(file)
end

जब हम इस कार्यक्रम को चलाते हैं, तो यह हमें एक त्रुटि देगा। लेकिन वोafter यह सुनिश्चित करेगा कि फ़ाइल डिस्क्रिप्टर ऐसी किसी भी घटना पर बंद हो।