अन्य उत्तर पूरी तरह से सही हैं, लेकिन यह उत्तर कुछ अतिरिक्त डिटेल प्रदान करता है, मुझे लगता है।
इस उदाहरण पर विचार करें:
using System;
static class Program {
static void Main() {
try {
ThrowTest();
} catch (Exception e) {
Console.WriteLine("Your stack trace:");
Console.WriteLine(e.StackTrace);
Console.WriteLine();
if (e.InnerException == null) {
Console.WriteLine("No inner exception.");
} else {
Console.WriteLine("Stack trace of your inner exception:");
Console.WriteLine(e.InnerException.StackTrace);
}
}
}
static void ThrowTest() {
decimal a = 1m;
decimal b = 0m;
try {
Mult(a, b); // line 34
Div(a, b); // line 35
Mult(b, a); // line 36
Div(b, a); // line 37
} catch (ArithmeticException arithExc) {
Console.WriteLine("Handling a {0}.", arithExc.GetType().Name);
// uncomment EITHER
//throw arithExc;
// OR
//throw;
// OR
//throw new Exception("We handled and wrapped your exception", arithExc);
}
}
static void Mult(decimal x, decimal y) {
decimal.Multiply(x, y);
}
static void Div(decimal x, decimal y) {
decimal.Divide(x, y);
}
}
यदि आप throw arithExc;
लाइन को अनलिमट करते हैं , तो आपका आउटपुट है:
Handling a DivideByZeroException.
Your stack trace:
at Program.ThrowTest() in c:\somepath\Program.cs:line 44
at Program.Main() in c:\somepath\Program.cs:line 9
No inner exception.
निश्चित रूप से, आपने जानकारी खो दी है कि यह अपवाद कहां हुआ है। यदि इसके बजाय आप throw;
लाइन का उपयोग करते हैं , तो आपको यही मिलता है:
Handling a DivideByZeroException.
Your stack trace:
at System.Decimal.FCallDivide(Decimal& d1, Decimal& d2)
at System.Decimal.Divide(Decimal d1, Decimal d2)
at Program.Div(Decimal x, Decimal y) in c:\somepath\Program.cs:line 58
at Program.ThrowTest() in c:\somepath\Program.cs:line 46
at Program.Main() in c:\somepath\Program.cs:line 9
No inner exception.
यह बहुत बेहतर है, क्योंकि अब आप देखते हैं कि यह वह Program.Div
तरीका था जिससे आपको परेशानी हुई। लेकिन यह देखना अभी भी मुश्किल है कि यह समस्या try
ब्लॉक में लाइन 35 या लाइन 37 से आती है या नहीं ।
यदि आप तीसरे विकल्प का उपयोग करते हैं, तो बाहरी अपवाद में लपेटकर, आप कोई जानकारी नहीं खोते हैं:
Handling a DivideByZeroException.
Your stack trace:
at Program.ThrowTest() in c:\somepath\Program.cs:line 48
at Program.Main() in c:\somepath\Program.cs:line 9
Stack trace of your inner exception:
at System.Decimal.FCallDivide(Decimal& d1, Decimal& d2)
at System.Decimal.Divide(Decimal d1, Decimal d2)
at Program.Div(Decimal x, Decimal y) in c:\somepath\Program.cs:line 58
at Program.ThrowTest() in c:\somepath\Program.cs:line 35
विशेष रूप से आप देख सकते हैं कि यह लाइन 35 है जो समस्या की ओर ले जाती है। हालाँकि, इसके लिए लोगों को खोज करने की आवश्यकता होती है InnerException
, और यह साधारण मामलों में आंतरिक अपवादों का उपयोग करने के लिए कुछ हद तक अप्रत्यक्ष लगता है।
में इस ब्लॉग पोस्ट वे बुला (प्रतिबिंब के माध्यम से) से लाइन नंबर (कोशिश ब्लॉक की लाइन) को बचाए रखने internal
intance विधि InternalPreserveStackTrace()
पर Exception
वस्तु। लेकिन इस तरह प्रतिबिंब का उपयोग करना अच्छा नहीं है (.NET फ्रेमवर्क अपने internal
सदस्यों को किसी दिन चेतावनी के बिना बदल सकता है )।