रूबी - अपवाद
निष्पादन और अपवाद हमेशा एक साथ चलते हैं। यदि आप एक फ़ाइल खोल रहे हैं, जो मौजूद नहीं है, तो यदि आपने इस स्थिति को ठीक से नहीं संभाला है, तो आपका प्रोग्राम खराब गुणवत्ता का माना जाता है।
अपवाद होने पर प्रोग्राम बंद हो जाता है। इसलिए विभिन्न प्रकार की त्रुटियों को संभालने के लिए अपवादों का उपयोग किया जाता है, जो एक कार्यक्रम के निष्पादन के दौरान हो सकता है और पूरी तरह से कार्यक्रम को रोकने के बजाय उचित कार्रवाई कर सकता है।
रूबी अपवादों को संभालने के लिए एक अच्छा तंत्र प्रदान करती हैं। हम उस कोड को संलग्न करते हैं जो एक शुरुआत / अंत ब्लॉक में एक अपवाद उठा सकता है और रूबी को यह बताने के लिए बचाव खंड का उपयोग कर सकता है कि हम किस प्रकार के अपवादों को संभालना चाहते हैं।
वाक्य - विन्यास
begin
# -
rescue OneTypeOfException
# -
rescue AnotherTypeOfException
# -
else
# Other exceptions
ensure
# Always will be executed
end
शुरू से बचाव तक सब कुछ संरक्षित है। यदि कोड के इस ब्लॉक के निष्पादन के दौरान कोई अपवाद होता है, तो नियंत्रण बचाव और अंत के बीच के ब्लॉक में जाता है ।
प्रत्येक के लिए बचाव में खंड शुरू ब्लॉक, रूबी बदले में मानकों से प्रत्येक के खिलाफ उठाया अपवाद है। मैच सफल होगा यदि बचाव खंड में नामित अपवाद वर्तमान में फेंके गए अपवाद के प्रकार के समान है, या उस अपवाद का सुपरक्लास है।
इस घटना में कि कोई अपवाद निर्दिष्ट किसी भी प्रकार की त्रुटि से मेल नहीं खाता है, हमें सभी बचाव खंडों के बाद किसी अन्य खंड का उपयोग करने की अनुमति है ।
उदाहरण
#!/usr/bin/ruby
begin
file = open("/unexistant_file")
if file
puts "File opened successfully"
end
rescue
file = STDIN
end
print file, "==", STDIN, "\n"
यह निम्नलिखित परिणाम का उत्पादन करेगा। आप देख सकते हैं कि एसटीडीआईएन फाइल करने के लिए प्रतिस्थापित किया गया है क्योंकि ओपन फेल है।
#<IO:0xb7d16f84>==#<IO:0xb7d16f84>
रिट्री स्टेटमेंट का उपयोग करना
आप बचाव ब्लॉक का उपयोग करके एक अपवाद पर कब्जा कर सकते हैं और फिर शुरू से ब्लॉक को निष्पादित करने के लिए रिट्री स्टेटमेंट का उपयोग कर सकते हैं ।
वाक्य - विन्यास
begin
# Exceptions raised by this code will
# be caught by the following rescue clause
rescue
# This block will capture all types of exceptions
retry # This will move control to the beginning of begin
end
उदाहरण
#!/usr/bin/ruby
begin
file = open("/unexistant_file")
if file
puts "File opened successfully"
end
rescue
fname = "existant_file"
retry
end
निम्नलिखित प्रक्रिया का प्रवाह है -
- एक अपवाद खुला हुआ।
- बचाव के लिए गया। fname फिर से सौंपा गया था।
- रिट्री द्वारा शुरुआत की शुरुआत में चला गया।
- यह समय फ़ाइल सफलतापूर्वक खुलती है।
- आवश्यक प्रक्रिया जारी रखी।
NOTE- ध्यान दें कि यदि पुन: प्रतिस्थापित नाम की फाइल मौजूद नहीं है, तो यह उदाहरण कोड असीम रूप से पुनर्प्रयास करता है। यदि आप अपवाद प्रक्रिया के लिए पुनः प्रयास का उपयोग करते हैं तो सावधान रहें ।
उठाना स्टेटमेंट का उपयोग करना
आप एक अपवाद को बढ़ाने के लिए स्टेटमेंट का उपयोग कर सकते हैं । जब भी यह कहा जाता है निम्नलिखित विधि एक अपवाद उठाती है। यह दूसरा संदेश छपा होगा।
वाक्य - विन्यास
raise
OR
raise "Error Message"
OR
raise ExceptionType, "Error Message"
OR
raise ExceptionType, "Error Message" condition
पहला रूप केवल वर्तमान अपवाद (या कोई वर्तमान अपवाद नहीं है, तो एक रंटइमार्ट) को फिर से उठाता है। इसका उपयोग अपवाद संचालकों में किया जाता है जिन्हें इसे पारित करने से पहले अपवाद को रोकना पड़ता है।
दूसरा रूप एक नया RuntimeError अपवाद बनाता है, इसके संदेश को दिए गए स्ट्रिंग में सेट करता है। यह अपवाद तब कॉल स्टैक उठाया जाता है।
तीसरा फ़ॉर्म अपवाद बनाने के लिए पहले तर्क का उपयोग करता है और फिर संबंधित संदेश को दूसरे तर्क पर सेट करता है।
चौथा रूप तीसरे रूप के समान है लेकिन आप कोई भी सशर्त विवरण जोड़ सकते हैं जब तक कि कोई अपवाद न उठाया जाए ।
उदाहरण
#!/usr/bin/ruby
begin
puts 'I am before the raise.'
raise 'An error has occurred.'
puts 'I am after the raise.'
rescue
puts 'I am rescued.'
end
puts 'I am after the begin block.'
यह निम्नलिखित परिणाम का उत्पादन करेगा -
I am before the raise.
I am rescued.
I am after the begin block.
एक और उदाहरण है जो उठाने के उपयोग को दर्शाता है -
#!/usr/bin/ruby
begin
raise 'A test exception.'
rescue Exception => e
puts e.message
puts e.backtrace.inspect
end
यह निम्नलिखित परिणाम का उत्पादन करेगा -
A test exception.
["main.rb:4"]
सुनिश्चित कथन का उपयोग करना
कभी-कभी, आपको यह गारंटी देने की आवश्यकता होती है कि कोड के एक ब्लॉक के अंत में कुछ प्रसंस्करण किया जाता है, चाहे एक अपवाद उठाया गया हो। उदाहरण के लिए, आपके पास ब्लॉक में प्रवेश पर एक फाइल खुली हो सकती है और ब्लॉक से बाहर निकलते ही आपको यह सुनिश्चित करना होगा कि यह बंद हो जाए।
यह सुनिश्चित करता है कि क्लॉज सिर्फ यही करता है। सुनिश्चित करें कि अंतिम बचाव खंड के बाद चला जाता है और कोड का एक हिस्सा होता है जिसे हमेशा ब्लॉक टर्मिनेशन के रूप में निष्पादित किया जाएगा। इससे कोई फर्क नहीं पड़ता कि ब्लॉक सामान्य रूप से बाहर निकलता है, अगर यह एक अपवाद को उठाता है या बचाता है, या यदि यह किसी अपवाद के अपवाद से समाप्त हो जाता है, तो सुनिश्चित करें कि ब्लॉक चल जाएगा।
वाक्य - विन्यास
begin
#.. process
#..raise exception
rescue
#.. handle error
ensure
#.. finally ensure execution
#.. This will always execute.
end
उदाहरण
begin
raise 'A test exception.'
rescue Exception => e
puts e.message
puts e.backtrace.inspect
ensure
puts "Ensuring execution"
end
यह निम्नलिखित परिणाम का उत्पादन करेगा -
A test exception.
["main.rb:4"]
Ensuring execution
कथन का उपयोग करना
यदि दूसरा खंड मौजूद है, तो यह बचाव खंड के बाद और किसी भी सुनिश्चित से पहले जाता है ।
कोड के मुख्य निकाय द्वारा कोई अपवाद नहीं उठाए जाने पर ही किसी अन्य खंड के निकाय को निष्पादित किया जाता है।
वाक्य - विन्यास
begin
#.. process
#..raise exception
rescue
# .. handle error
else
#.. executes if there is no exception
ensure
#.. finally ensure execution
#.. This will always execute.
end
उदाहरण
begin
# raise 'A test exception.'
puts "I'm not raising exception"
rescue Exception => e
puts e.message
puts e.backtrace.inspect
else
puts "Congratulations-- no errors!"
ensure
puts "Ensuring execution"
end
यह निम्नलिखित परिणाम का उत्पादन करेगा -
I'm not raising exception
Congratulations-- no errors!
Ensuring execution
उठाया त्रुटि संदेश $ का उपयोग करके कब्जा किया जा सकता है! चर।
कैच एंड थ्रो
हालांकि, उठान और बचाव का अपवाद तंत्र निष्पादन को छोड़ने के लिए बहुत अच्छा है जब चीजें गलत हो जाती हैं, तो कभी-कभी सामान्य प्रसंस्करण के दौरान कुछ गहरी नेस्टेड निर्माण से बाहर निकलने में सक्षम होना अच्छा होता है। यह वह जगह है जहां पकड़ और फेंक काम में आते हैं।
पकड़ एक ब्लॉक है कि व्यक्ति का नाम (जो एक प्रतीक या एक स्ट्रिंग हो सकता है) के साथ लेबल है परिभाषित करता है। ब्लॉक को सामान्य रूप से तब तक निष्पादित किया जाता है जब तक कि एक थ्रो का सामना नहीं किया जाता है।
वाक्य - विन्यास
throw :lablename
#.. this will not be executed
catch :lablename do
#.. matching catch will be executed after a throw is encountered.
end
OR
throw :lablename condition
#.. this will not be executed
catch :lablename do
#.. matching catch will be executed after a throw is encountered.
end
उदाहरण
निम्न उदाहरण उपयोगकर्ता के साथ बातचीत को समाप्त करने के लिए थ्रो का उपयोग करता है यदि '!' किसी भी संकेत के जवाब में टाइप किया गया है।
def promptAndGet(prompt)
print prompt
res = readline.chomp
throw :quitRequested if res == "!"
return res
end
catch :quitRequested do
name = promptAndGet("Name: ")
age = promptAndGet("Age: ")
sex = promptAndGet("Sex: ")
# ..
# process information
end
promptAndGet("Name:")
आपको अपने मशीन पर उपरोक्त कार्यक्रम की कोशिश करनी चाहिए क्योंकि इसे मैन्युअल इंटरैक्शन की आवश्यकता है। यह निम्नलिखित परिणाम का उत्पादन करेगा -
Name: Ruby on Rails
Age: 3
Sex: !
Name:Just Ruby
कक्षा का अपवाद
रूबी के मानक वर्ग और मॉड्यूल अपवादों को बढ़ाते हैं। सभी अपवाद वर्ग शीर्ष पर वर्ग अपवाद के साथ एक पदानुक्रम बनाते हैं। अगले स्तर में सात अलग-अलग प्रकार हैं -
- Interrupt
- NoMemoryError
- SignalException
- ScriptError
- StandardError
- SystemExit
इस स्तर पर एक अन्य अपवाद है, Fatal, लेकिन रूबी दुभाषिया केवल आंतरिक रूप से इसका उपयोग करता है।
ScriptError और StandardError दोनों में कई उपवर्ग हैं, लेकिन हमें यहाँ विवरण में जाने की आवश्यकता नहीं है। महत्वपूर्ण बात यह है कि यदि हम अपने अपवाद वर्ग बनाते हैं, तो उन्हें कक्षा अपवाद या उसके किसी वंशज के उपवर्ग होने चाहिए।
आइए एक उदाहरण देखें -
class FileSaveError < StandardError
attr_reader :reason
def initialize(reason)
@reason = reason
end
end
अब, निम्नलिखित उदाहरण को देखें, जो इस अपवाद का उपयोग करेगा -
File.open(path, "w") do |file|
begin
# Write out the data ...
rescue
# Something went wrong!
raise FileSaveError.new($!)
end
end
यहाँ महत्वपूर्ण लाइन FileSaveError.new ($!) है । हम कॉल करने के लिए संकेत देते हैं कि एक अपवाद उत्पन्न हुआ है, इसे फाइलशेयर के एक नए उदाहरण से गुजरते हुए, इस कारण से कि विशिष्ट अपवाद के कारण डेटा का लेखन विफल हो गया।