Exception Filters in C#
Understanding Exception Filters in C#
**Exception Filters** allow handling exceptions **conditionally** before executing a `catch` block. Introduced in **C# 6.0**, they improve **code readability** and **exception handling efficiency**.
Key Features of Exception Filters
- Allows filtering **specific exception scenarios** before entering a `catch` block.
- Improves **code readability** by avoiding nested `if` statements in `catch` blocks.
- Prevents **unnecessary exception handling** when a condition isn't met.
- Works well with **logging exceptions without catching them**.
Basic Usage of Exception Filters
Exception filters use the `when` keyword to specify **conditions** for handling exceptions.
Example: Using Exception Filters to Handle Specific Cases
using System;
class Program
{
static void Main()
{
try
{
ThrowException(404);
}
catch (Exception ex) when (ex.Message.Contains("404"))
{
Console.WriteLine("Not Found Error Handled.");
}
catch (Exception ex)
{
Console.WriteLine($"General Exception: {ex.Message}");
}
}
static void ThrowException(int errorCode)
{
if (errorCode == 404)
throw new Exception("404 - Not Found");
else
throw new Exception("500 - Internal Server Error");
}
}
// Output:
// Not Found Error Handled.
The `when (ex.Message.Contains("404"))` filter ensures that only `404` errors enter the first `catch` block.
Using Exception Filters for Logging Without Handling
Exception filters allow logging an exception **without catching it**, ensuring normal exception propagation.
Example: Logging Errors Using Exception Filters
using System;
using System.IO;
class Program
{
static void Main()
{
try
{
ThrowError();
}
catch (Exception ex) when (LogError(ex))
{
// This block is not executed, but exception is logged.
}
}
static void ThrowError()
{
throw new Exception("An unexpected error occurred.");
}
static bool LogError(Exception ex)
{
File.WriteAllText("error.log", $"Logged Error: {ex.Message}");
Console.WriteLine("Error logged.");
return false; // Allows exception to propagate
}
}
// Output:
// Error logged.
// (Exception still propagates and crashes the program)
The **when (LogError(ex))** filter logs the exception **without handling it**, allowing normal exception propagation.
Combining Multiple Exception Filters
Exception filters can be **stacked** to handle different scenarios efficiently.
Example: Using Multiple Filters for Different Error Types
using System;
class Program
{
static void Main()
{
try
{
ThrowCustomError(401);
}
catch (Exception ex) when (ex.Message.Contains("401"))
{
Console.WriteLine("Unauthorized Access Handled.");
}
catch (Exception ex) when (ex.Message.Contains("500"))
{
Console.WriteLine("Internal Server Error Handled.");
}
catch (Exception ex)
{
Console.WriteLine($"Unhandled Exception: {ex.Message}");
}
}
static void ThrowCustomError(int code)
{
if (code == 401) throw new Exception("401 - Unauthorized");
if (code == 500) throw new Exception("500 - Server Error");
throw new Exception("Unknown Error");
}
}
// Output (for different codes):
// Unauthorized Access Handled. (if code is 401)
// Internal Server Error Handled. (if code is 500)
// Unhandled Exception: Unknown Error (for others)
Different `when` conditions handle **different error codes**, reducing the need for multiple `if` checks inside a single `catch` block.
Exception Filters vs Traditional `catch` Blocks
Feature | Traditional `catch` | Exception Filters (`when`) |
---|---|---|
Handles exceptions conditionally | ❌ No | ✅ Yes |
Prevents unnecessary handling | ❌ No | ✅ Yes |
Can log errors without catching | ❌ No | ✅ Yes |
Requires additional `if` checks | ✅ Yes | ❌ No |
Best Practices for Using Exception Filters
- Use exception filters **to handle specific cases before catching exceptions**.
- Use `when (LogError(ex))` to **log exceptions without handling them**.
- Avoid **complex logic inside `when` filters**—keep conditions simple.
- Use **multiple filters** instead of `if` checks inside `catch` blocks.