Author avatar

Peter Mbanugo

Throw and Re-throw Exceptions in C#

Peter Mbanugo

  • Oct 18, 2018
  • 7 Min read
  • 100,179 Views
  • Oct 18, 2018
  • 7 Min read
  • 100,179 Views
C#

Introduction

Writing software is a complex undertaking, and it is quite common for even the best software to ship with various problems. Sometimes the problem is caused by bad code; other times, a problem is caused by bad user input that has not been accounted for in the application’s code. Regardless of the cause of the problem, the end result is that the application does not work as expected. At this point, we say the application has encountered an error. In .NET, exceptions are thrown when a running application encounters an error.

What Are Exceptions?

An exception is a runtime error in a program that violates a system or application constraint, or a condition that is not expected to occur during normal execution of the program. Possible exceptions include attempting to connect to a database that no longer exists when a program tries to divide a number by zero or opening a corrupted XML file. When these occur, the system catches the error and raises an exception.

Throwing Exceptions

When a situation occurs that violates a system or application constraint, it can generate an exception to signal to the caller that an operation has failed. The process of generating and signaling the error is referred to as throwing exception. This is done using the throw keyword followed by a new instance of a class deriving from System.Exception. Let's look at an example. Create a new console project and update Program.cs with the code below.

1using System;
2
3namespace MyApp
4{
5    class Program
6    {
7        static void Main(string[] args)
8        {
9            var radio = new Radio();
10            radio.SetVolume(120);
11        }
12
13        class Radio
14        {
15            public int Volume { get; set; }
16            public string Station { get; set; }
17
18            public void SetVolume(int volume)
19            {
20                if (volume > 100)
21                {
22                    throw new ArgumentOutOfRangeException(nameof(volume), "volume cannot be more than 100");
23                }
24
25                Volume = volume;
26            }
27
28            public void SetStation(string station)
29            {
30                if (string.IsNullOrEmpty(station))
31                {
32                    throw new ArgumentNullException(nameof(station), "you cannot tune to an empty station");
33                }
34
35                Station = station;
36            }
37        }
38    }
39}
csharp

In the code example above, we defined a Radio class, the properties Station and Volume, and methods SetStation and SetVolume. When the SetVolume method is called with a value greater than 100, the application throws an ArgumentOutOfRangeException with the parameter name and exception message as arguments. Similarly, when SetStation method is called with an empty string, it throws an ArgumentNullException with the parameter name and exception message as arguments.

If we run the application and it crashes, it will print out the exception message and stack trace tells us where the error occurred in the console. The information that it'll give you should be similar to what is displayed in the image below:

throw exception.png

From the stack trace information, you can see that it points to the line where we used the throw keyword, followed by the line that called the SetVolume method. This information is very helpful when debugging.

Re-throwing Exceptions

The exception from the previous section propagates up the call stack and since we have no code to catch and handle the exception, it crashes. When an exception is caught, we can perform some operations, like logging the error, and then re-throw the exception. Re-throwing an exception means calling the throw statement without an exception object, inside a catch block. It can only be used inside a catch block.

Let's create a new console application and update the Program.cs file with the following:

1using System;
2
3namespace MyApp
4{
5    class Program
6    {
7        static void Main(string[] args)
8        {
9            var calculator = new Calculator();
10
11            Console.WriteLine("Enter number");
12            int number = int.Parse(Console.ReadLine());
13
14            Console.WriteLine("Enter divisor");
15            int divisor = int.Parse(Console.ReadLine());
16
17            Console.WriteLine(calculator.Divide(number, divisor));
18        }
19
20        class Calculator
21        {
22            public int Divide(int number, int divisor)
23            {
24                try
25                {
26                    return number / divisor;
27                }
28                catch (DivideByZeroException)
29                {
30                    //TODO: log error
31                    Console.WriteLine("Can't divide by 0");
32                    throw;//propage this error
33                }
34
35            }
36        }
37    }
38}
csharp

From the code above, we have a Calculator class with Divide method. In .NET, when a number is being divided by 0, it throws the DivideByZeroException. In the Divide method, we have code to catch this exception, log to the console, and re-throw the exception. Run the application and enter a divisor of 0:

re-throw exception.gif

You can see that when we passed it 0 as a divisor, it printed Can't divide by 0 to the console before re-throwing the exception. Notice that the stack trace still maintained proper information, pointing to line 26 as the line that the error occurred at, even though it was re-thrown on line 32. A common mistake people make when they intend to re-throw exception is to call the throw keyword with the caught exception object. An example of this is as follows:

1catch (DivideByZeroException ex)
2{
3    //TODO: log error
4    Console.WriteLine("Can't divide by 0");
5    throw ex;//propage this error
6}
csharp

If we had it implemented this way, it'll print out the following stack trace:

1Unhandled Exception: System.DivideByZeroException: Attempted to divide by zero.
2   at MyApp.Program.Calculator.Divide(Int32 number, Int32 divisor) in /Users/pmbanugo/Documents/projects/dotnet/MyApp/Program.cs:line 32
3   at MyApp.Program.Main(String[] args) in /Users/pmbanugo/Documents/projects/dotnet/MyApp/Program.cs:line 17

You can see that it pointed to line 32, the line where the exception was thrown, rather than where it occurred. Re-throwing an exception is simply calling throw without an exception object.

That's a Wrap

Sometimes an error occurs in our application because of bad code, other times, it is caused by bad user input that has not been accounted for in the application’s code. An exception is thrown when an error is encountered in a running application. In this guide we looked at using the throw keyword for throwing and re-throwing exception and explained the correct syntax for re-throwing exception.