Exception Class Hierarchy in C#
Understanding Exception Class Hierarchy in C#
In C#, all exceptions are derived from the System.Exception
class. The hierarchy helps categorize exceptions into different types, making it easier to handle specific errors.
Exception Class Hierarchy
The exception hierarchy in C# follows this structure:
System.Object
└── System.Exception
├── System.SystemException
│ ├── System.IndexOutOfRangeException
│ ├── System.NullReferenceException
│ ├── System.InvalidOperationException
│ ├── System.StackOverflowException
│ ├── System.OutOfMemoryException
│ ├── System.DivideByZeroException
│ ├── System.FormatException
│ └── System.IO.IOException
├── System.ApplicationException (Base for custom exceptions)
│ ├── CustomException1
│ └── CustomException2
└── Other Exception Classes
The System.Exception
class serves as the base class for all exceptions, with System.SystemException
handling runtime errors and System.ApplicationException
allowing custom exceptions.
Common Built-in Exception Types
Here are some commonly used exceptions in C#:
Exception Type | Derived From | Description | Example |
---|---|---|---|
IndexOutOfRangeException |
SystemException |
Thrown when accessing an invalid array index. | arr[10] (when arr length is 5) |
NullReferenceException |
SystemException |
Thrown when trying to access an object that is null. | string s = null; s.Length; |
DivideByZeroException |
SystemException |
Thrown when dividing by zero. | int x = 5 / 0; |
InvalidOperationException |
SystemException |
Thrown when an operation is not allowed. | var item = list.First(); (when list is empty) |
IOException |
SystemException |
Thrown when an I/O operation fails. | File not found, disk full, etc. |
Creating Custom Exceptions
You can define your own exception by deriving from ApplicationException
or Exception
.
Example: Creating a Custom Exception
using System;
public class InvalidAgeException : ApplicationException
{
public InvalidAgeException(string message) : base(message) { }
}
// Usage
class Program
{
static void Main()
{
int age = 15;
try
{
if (age < 18)
throw new InvalidAgeException("Age must be 18 or older.");
}
catch (InvalidAgeException ex)
{
Console.WriteLine($"Custom Exception Caught: {ex.Message}");
}
}
}
// Output:
// Custom Exception Caught: Age must be 18 or older.
This custom exception checks if an age is below 18 and throws an error accordingly.
Handling Different Exception Types
You can handle multiple exception types using multiple catch
blocks.
Example: Handling Multiple Exception Types
using System;
class Program
{
static void Main()
{
try
{
int[] arr = new int[3];
Console.WriteLine(arr[5]); // Will throw IndexOutOfRangeException
}
catch (IndexOutOfRangeException ex)
{
Console.WriteLine($"Index Error: {ex.Message}");
}
catch (Exception ex)
{
Console.WriteLine($"General Error: {ex.Message}");
}
}
}
// Output:
// Index Error: Index was outside the bounds of the array.
The catch
block for IndexOutOfRangeException
executes first, as it is more specific.
When to Use Custom Exceptions?
- When built-in exceptions do not describe your error properly.
- To enforce business logic validation (e.g., "Age must be 18 or older").
- To improve debugging and maintainability.
- To categorize errors in large applications.