} catch (DivideByZeroException ex) {
Console.WriteLine("Division by zero error.");
} catch (ArithmeticException ex) {
Console.WriteLine("Numerator cannot be zero.");
} catch (FormatException ex) {
Console.WriteLine("Input error.");
} catch (Exception ex) {
Console.WriteLine(ex.Message);
}
Console.ReadLine();
}
The following shows the output when different values are entered for num1
and num2
:
Please enter the first number:5
Please enter the second number:0
Division by zero error.
Please enter the first number:0
Please enter the second number:5
Numerator cannot be zero.
Please enter the first number:a
Input error.
There are times when after catching an exception, you want to throw the same (or a new type) exception back to the calling function after taking some corrective actions. Take a look at this example:
class Program {
static void Main(string[] args) {
int num1, num2, result;
try {
Console.Write("Please enter the first number:");
num1 = int.Parse(Console.ReadLine());
Console.Write("Please enter the second number:");
num2 = int.Parse(Console.ReadLine());
Program myApp = new Program();
Console.WriteLine("The result of {0}/{1} is {2}", num1, num2,
myApp.PerformDivision(num1, num2));
} catch (Exception ex) {
Console.WriteLine(ex.Message);
if (ex.InnerException != null)
Console.WriteLine(ex.InnerException.ToString());
}
Console.ReadLine();
}
private int PerformDivision(int num1, int num2) {
try {
return num1 / num2;
} catch (DivideByZeroException ex) {
throw new Exception("Division by zero error.", ex);
}
}
}
Here, the PerformDivision()
function tries to catch the DivideByZeroException
exception and once it succeeds, it rethrows a new generic Exception
exception, using the following statements with two arguments:
throw new Exception("Division by zero error.", ex);
The first argument indicates the description for the exception to be thrown, while the second argument is for the inner exception. The inner exception indicates the exception that causes the current exception. When this exception is rethrown, it is handled by the catch block in the Main()
function:
catch (Exception ex) {
Console.WriteLine(ex.Message);
if (ex.InnerException != null)
Console.WriteLine(ex.InnerException.ToString());
}
To retrieve the source of the exception, you can check the InnerException
property and print out its details using the ToString()
method. Here's the output when num2 is zero:
Please enter the first number:5
Please enter the second number:0
Division by zero error.
System.DivideByZeroException: Attempted to divide by zero.
at ConsoleApp.Program.PerformDivision(Int32 num1, Int32 num2) in C:\Documents and Settings\Wei-Meng Lee\My Documents\Visual Studio 2008\Projects\ConsoleApp\ConsoleApp\Program.cs:line 43
As you can see, the message of the exception is "Division by zero error" (set by yourself) and the InnerException
property shows the real cause of the error — "Attempted to divide by zero."
The InnerException
property is of type Exception
, and it can be used to store a list of previous exceptions. This is known as exception chaining.
To see how exception chaining works, consider the following program:
class Program {
static void Main(string[] args) {
Program myApp = new Program();
try {
myApp.Method1();
} catch (Exception ex) {
Console.WriteLine(ex.Message);
if (ex.InnerException != null)
Console.WriteLine(ex.InnerException.ToString());
}
Console.ReadLine();
}
private void Method1() {
try {
Method2();
} catch (Exception ex) {
throw new Exception(
"Exception caused by calling Method2() in Method1().", ex);
}
}
private void Method2() {
try {
Method3();
} catch (Exception ex) {
throw new Exception(
"Exception caused by calling Method3() in Method2().", ex);
}
}
private void Method3() {
try {
int num1 = 5, num2 = 0;
int result = num1 / num2;
} catch (DivideByZeroException ex) {
throw new Exception("Division by zero error in Method3().", ex);
}
}
}
In this program, the Main()
function calls Method1()
, which in turns calls Method2()
. Method2()
then calls Method3()
. In Method3(),
a division-by-zero exception occurs and you rethrow a new Exception
exception by passing in the current exception ( DividebyZeroException
). This exception is caught by Method2()
, which rethrows a new Exception
exception by passing in the current exception. Method1()
in turn catches the exception and rethrows a new Exception
exception. Finally, the Main()
function catches the exception and prints out the result as shown in Figure 12-4.
Figure 12-4
If you set a breakpoint in the catch block within the Main()
function, you will see that the InnerException
property contains details of each exception and that all the exceptions are chained via the InnerException
property (see Figure 12-5).
Figure 12-5
Instead of using the default description for each exception class you are throwing, you can customize the description of the exception by creating an instance of the exception and then setting the Message
property. You can also specify the HelpLink
property to point to a URL where developers can find more information about the exception. For example, you can create a new instance of the ArithmeticException
class using the following code:
Читать дальше