异常过滤器通过when关键字按条件捕获异常,示例中根据异常消息、HResult或租户策略判断,避免重抛导致的性能损耗与堆栈丢失,适用于日志分级、临时故障重试等场景。

在 C# 中,异常过滤器(Exception Filters)让你能在 catch 块中更精确地决定是否处理某个异常,而不是简单地根据异常类型来捕获。它通过 when 关键字实现,只有当指定的条件为 true 时,才会真正进入该 catch 块。
使用 when 关键字进行条件过滤
异常过滤器允许你在 catch 后添加一个布尔表达式,.NET 会先判断这个表达式,再决定是否执行 catch 块。
示例:try
{
throw new InvalidOperationException("网络错误");
}
catch (InvalidOperationException ex) when (ex.Message.Contains("网络"))
{
Console.WriteLine("捕获到网络相关的操作异常");
}
catch (InvalidOperationException ex)
{
Console.WriteLine("其他操作异常");
}上面代码中,第一个 catch 只有在异常消息包含“网络”时才会触发,否则跳过并尝试下一个匹配的 catch 块。
基于异常属性或上下文做判断
你可以根据异常的具体属性(如 HResult、Source、自定义字段)或外部环境(如当前用户、配置项)来决定是否处理。
示例:根据 HResult 过滤 COM 异常try
{
// 调用 COM 组件可能抛出此类异常
throw new COMException("访问被拒绝", unchecked((int)0x80070005));
}
catch (COMException ex) when (ex.HResult == unchecked(0x80070005))
{
Console.WriteLine("捕获特定 HResult 的 COM 异常:权限不足");
}避免异常重新抛出带来的性能损耗
传统做法中,为了区分不同情况,可能会先 catch 再判断然后重新 throw,这会影响堆栈信息和性能。异常过滤器不会破坏原始堆栈,且不被视为“捕获-重抛”模式。
对比说明:- 传统方式:catch 后判断,不符合条件再 throw,已发生堆栈截断
- 过滤器方式:只有 when 条件为 true 才真正捕获,堆栈保持完整
实际应用场景建议
异常过滤器适合用于以下场景:
- 日志系统中,仅对特定错误码记录严重级别
- 分布式调用中,只重试某些临时性故障(如超时、限流)
- 多租户系统中,根据不同租户策略处理异常
string currentTenant = "beta-user";try { throw new ServiceException("服务不可用"); } catch (ServiceException ex) when (currentTenant == "beta-user" && IsTransientError(ex)) { Console.WriteLine("对灰度用户启用自动重试逻辑"); } catch (ServiceException ex) { Console.WriteLine("普通用户直接报错"); }
基本上就这些。异常过滤器让异常处理更灵活、高效,尤其适合需要精细化控制的场景,同时保留了完整的异常上下文信息。合理使用能显著提升代码的可维护性和健壮性。不复杂但容易忽略。








