The Exception Filters feature in C# 6.0 provides various benefits. Here’s an explanation of some (ordered by my perceived importance)
-
Feature Parity – Exception filters were already implemented in the IL level and the other .Net languages (VB.Net & F#)[1] and as part of building the new compiler for C# and VB.Net (project “Roslyn”) many features existing in one language and lacking in the other were implemented[2].
-
Crash Dumps – Exception filters don’t modify the stack. which means that if it gets dumped (in a crash dump) you would be able to know where the exception was originally thrown and not only where it was rethrown (which is irrelevant to the actual problem)[3]
-
Debugging – When an exception enters a
catch
block, rethrown usingthrow;
and isn’t handled anywhere else in the stack (and the exception settings are set to break when the exception is user-unhandled) the debugger would break onthrow;
instead of where the exception is originally thrown (i.e. in the example below it would break onthrow;
and notthrow new FileNotFoundException();
) -
Multiple
catch
Blocks – Without exception filters you must catch the exception, check a condition and if it isn’t metthrow;
the exception. The rethrown exception doesn’t consider any othercatch
blocks even if the exception satisfies all the conditions which ultimately results in a single bigcatch
blocktry { throw new FileNotFoundException(); } catch (FileNotFoundException e) { if (!e.FileName.Contains("Hamster")) { throw; } // Handle } catch (Exception e) { // Unreachable code }
-
Readability – While you could use a “catch all”
catch
block with many conditions andthrow;
when they are not met (while suffering the modification to the stack) it’s much clearer to have separate and distinctcatch
blocks where each handles a specific problem in the appropriate way:try { } catch (Win32Exception exception) when (exception.NativeErrorCode == 0x00042) { } catch (Win32Exception exception) when (exception.NativeErrorCode == 0x00011) { } catch (IOException) { }
-
“Abuse” – You could use exception filters as a way to inspect an exception without handling it. While this is not a main benefit it’s a nice side effect. You can have a false-returning logging method and an empty
catch
block:private static bool LogException(Exception exception) { Console.WriteLine(exception.Message); return false; } try { } catch (ArgumentException exception) when (LogException(exception)) { // Unreachable code. }
In conclusion, most of C# 6.0 features are small improvements and syntactic sugar, and while exception filters isn’t a very big feature it does provide functionality that wasn’t possible before.
-
A C# 6.0 Language Preview
The other exception improvement in C# 6.0—support for exception filters—brings the language up-to-date with other .NET languages, namely Visual Basic .NET and F#
-
Languages features in C# 6 and VB 14
-
New Features in C# 6
Exception filters are preferable to catching and rethrowing because they leave the stack unharmed. If the exception later causes the stack to be dumped, you can see where it originally came from, rather than just the last place it was rethrown.