您的位置:首页 >c#如何自定义异常_c#自定义异常看这一篇就够了_保姆级教程
发布于2026-05-03 阅读(0)
扫一扫,手机访问

开门见山,先说一个最核心的结论:在C#里创建自定义异常,并不是一个宽泛的“必须继承Exception”的要求。准确来说,它**必须继承System.Exception,或者继承它的任意一个子类**。这是条硬性规定,违反的代价就是编译时直接报错CS0155: Cannot throw an object that does not inherit from 'System.Exception'。
这背后的原因,得追溯到C#异常机制的底层设计。在CLR(公共语言运行时)层面,它就硬性规定所有能被抛出的对象,其类型必须是System.Exception的派生类。你可以做个简单实验:写一个空类class MyErr { },然后尝试throw new MyErr();——编译器会立刻拒绝你。
Exception类是CLR唯一认可的“合法异常根类型”(它本身通过一些机制,确保了异常构造的逻辑是封闭的)。ApplicationException。这个类早已被微软标记为[Obsolete],官方文档也明确建议“不要再使用它”。ArgumentException或InvalidOperationException,而不是凭空去造一个新的根类。接下来是关键实操。在.NET Core/.NET 5+的时代,序列化(比如用于跨进程通信、WCF、gRPC场景)依然是绕不开的话题。如果忽略了序列化相关的构造函数,你的异常信息在远程调用或日志捕获时,很可能会丢失。
一个健壮的自定义异常,通常需要实现以下四个构造函数:
(string message)的构造函数(这是最常用的场景)。(string message, Exception innerException)的构造函数(用于构建异常链,保留内部异常信息)。(SerializationInfo info, StreamingContext context)构造函数(为了兼容.NET Framework,以及在.NET Core中满足显式序列化的需求)。来看一个标准示例:
public class InvalidOrderStateException : Exception
{
public InvalidOrderStateException() { }
public InvalidOrderStateException(string message) : base(message) { }
public InvalidOrderStateException(string message, Exception innerException)
: base(message, innerException) { }
protected InvalidOrderStateException(SerializationInfo info, StreamingContext context)
: base(info, context) { }
}
当然,很多时候光有错误信息还不够。为异常添加额外属性(比如ErrorCode、OrderId)非常有意义,但这里头有两点需要特别注意:
info参数中把值读回来,否则序列化再反序列化之后,属性值就会变成默认值(比如0或者null)。Exception.Data这个字典属性。它无需修改异常类的结构,并且会自动参与序列化过程。下面是一个安全地补充了ErrorCode属性的写法示例:
public class PaymentFailedException : Exception
{
public int ErrorCode { get; }
public PaymentFailedException(int errorCode, string message)
: base(message)
{
ErrorCode = errorCode;
}
protected PaymentFailedException(SerializationInfo info, StreamingContext context)
: base(info, context)
{
ErrorCode = info.GetInt32(nameof(ErrorCode));
}
public override void GetObjectData(SerializationInfo info, StreamingContext context)
{
info.AddValue(nameof(ErrorCode), ErrorCode);
base.GetObjectData(info, context);
}
}
最后是一些提升工程化水平的细节。特性[Serializable]在.NET Framework中是必须显式添加的;在.NET Core/.NET 5+里虽然不强制,但如果缺少它,会导致GetObjectData方法不会被调用,你的自定义序列化逻辑也就失效了。
Exception结尾,这是.NET的命名约定,也是Roslyn Analyzer等静态分析工具的检查项。话说回来,比记住所有构造函数写法更重要的,是理解异常的本质:**异常一旦被抛出,就标志着正常的程序流程已经中断。它的核心职责,是清晰、准确地传达“什么错了、在哪错的、以及为什么错”,而不是去修复问题或触发重试逻辑。** 把握住这个原则,很多设计上的纠结就迎刃而解了。
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
正版软件
正版软件
正版软件
正版软件
正版软件
1
2
3
7
9