当前位置: 首页
编程语言
c#如何自定义异常_c#自定义异常看这一篇就够了_保姆级教程

c#如何自定义异常_c#自定义异常看这一篇就够了_保姆级教程

热心网友 时间:2026-05-06
转载

C#自定义异常必须继承System.Exception或其子类,否则编译报错CS0155;需实现4个构造函数以支持序列化,推荐加ErrorCode等属性但须正确处理序列化,并遵循命名、注释和轻量原则。

c#如何自定义异常_c#自定义异常看这一篇就够了_保姆级教程

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

在C#开发中,自定义异常是处理特定业务错误的常用手段。但很多开发者存在一个误区:认为只要继承Exception类即可。实际上,C#自定义异常有更严格的规范:**它必须直接继承System.Exception,或者继承其任意一个子类**。这是CLR层面的硬性要求,违反此规则将直接导致编译错误CS0155: Cannot throw an object that does not inherit from 'System.Exception'

为什么自定义异常必须继承 System.Exception

这源于C#异常处理机制的底层设计。公共语言运行时(CLR)规定,所有可被throw关键字抛出的对象,其类型必须是System.Exception的派生类。你可以通过一个简单测试验证:创建一个空类class MyError { },然后尝试执行throw new MyError();,编译器会立即阻止这一操作。

  • 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) { }
}

如何为自定义异常添加额外属性

在许多业务场景中,仅靠错误信息不足以定位问题。为异常添加额外属性(如ErrorCodeOrderIdUserId)能极大提升调试效率。但需注意以下关键点:

  • 所有自定义属性必须在**每一个**构造函数中被正确初始化。特别是在反序列化构造函数中,必须从info参数中读取并还原属性值,否则序列化/反序列化后属性将变为默认值。
  • 如果附加信息仅用于辅助日志记录或临时调试,更推荐使用Exception.Data字典属性。它无需修改异常类结构,且会自动参与序列化过程。
  • 安全警告:切勿将敏感信息(如用户密码、API密钥、访问令牌)存入异常属性。异常对象可能被记录到明文日志或意外暴露给客户端,导致安全风险。

以下是一个安全添加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方法不被调用,自定义序列化逻辑失效。
  • 添加XML注释:这不仅提升代码可读性,更能在Visual Studio智能提示、IntelliSense及生成的API文档中直接显示,对团队协作与代码维护至关重要。
  • 命名规范:异常类名应以Exception结尾,这是.NET命名约定,也是Roslyn Analyzer等代码分析工具的检查项。
  • 保持轻量:避免在异常构造函数中执行耗时操作(如文件I/O、网络请求、复杂计算)。异常对象的创建应快速、无副作用。

最后,理解异常的本质比记住所有语法更重要:**异常一旦被抛出,即标志着正常的程序流程已中断。其核心职责是清晰、准确地传达“错误类型、发生位置及根本原因”,而非执行修复或重试逻辑。** 把握这一原则,你的异常设计将更加清晰、有效。

来源:https://www.php.cn/faq/2315390.html

游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

同类文章
更多
怎么利用 System.err 输出错误流并在控制台中以醒目的颜色标记(取决于终端)

怎么利用 System.err 输出错误流并在控制台中以醒目的颜色标记(取决于终端)

怎么利用 System err 输出错误流并在控制台中以醒目的颜色标记(取决于终端) System err 默认行为不带颜色,终端是否显示颜色取决于自身支持 首先得明确一点:System err 本质上只是 Ja va 标准库里的一个 PrintStream 对象。它本身并不负责“颜色”这种花哨的玩

时间:2026-05-06 09:59
如何在 Java 中使用 ThreadLocal.remove() 确保在线程池复用场景下不会发生数据污染

如何在 Java 中使用 ThreadLocal.remove() 确保在线程池复用场景下不会发生数据污染

如何在 Ja va 中使用 ThreadLocal remove() 确保在线程池复用场景下不会发生数据污染 说到线程池和 ThreadLocal 的搭配使用,一个看似不起眼、实则极易“踩坑”的细节就是数据清理。想象一下,你精心设计的线程池正在高效运转,却因为某个任务留下的“数据尾巴”,导致后续任务

时间:2026-05-06 09:59
怎么利用 Arrays.asList() 转换出的“受限列表”理解其对 add() 等修改操作的限制

怎么利用 Arrays.asList() 转换出的“受限列表”理解其对 add() 等修改操作的限制

Arrays asList():一个“受限”但实用的列表视图 在Ja va开发中,Arrays asList()是一个高频使用的方法,但你是否真正了解它返回的是什么?一个常见的误解是,它直接生成了一个标准的ArrayList。事实并非如此。 简单来说,Arrays asList()返回的并非我们熟悉

时间:2026-05-06 09:59
如何在 Java 中利用 try-catch 实现对“软错误”的平滑感知与非侵入式监控日志记录

如何在 Java 中利用 try-catch 实现对“软错误”的平滑感知与非侵入式监控日志记录

如何在 Ja va 中利用 try-catch 实现对“软错误”的平滑感知与非侵入式监控日志记录 在 Ja va 开发中,我们常常会遇到一些“软错误”——它们不会让程序直接崩溃,却可能悄悄影响业务的正确性或用户体验。比如,调用第三方 API 时返回了空响应、缓存查询未命中、配置文件里某个非关键项缺失

时间:2026-05-06 09:59
Django怎么防止Celery任务重复执行_Python结合Redis实现分布式锁

Django怎么防止Celery任务重复执行_Python结合Redis实现分布式锁

Django怎么防止Celery任务重复执行:Python结合Redis实现分布式锁 你遇到过吗?明明只发了一次任务,后台却执行了两次。这不是代码写错了,而是分布式环境下一个经典的老朋友:多个worker同时抢到了同一个活儿。 为什么Celery任务会重复执行 问题的根源在于竞争。想象一下,多个Ce

时间:2026-05-06 09:58
热门专题
更多
刀塔传奇破解版无限钻石下载大全 刀塔传奇破解版无限钻石下载大全
洛克王国正式正版手游下载安装大全 洛克王国正式正版手游下载安装大全
思美人手游下载专区 思美人手游下载专区
好玩的阿拉德之怒游戏下载合集 好玩的阿拉德之怒游戏下载合集
不思议迷宫手游下载合集 不思议迷宫手游下载合集
百宝袋汉化组游戏最新合集 百宝袋汉化组游戏最新合集
jsk游戏合集30款游戏大全 jsk游戏合集30款游戏大全
宾果消消消原版下载大全 宾果消消消原版下载大全
  • 日榜
  • 周榜
  • 月榜
热门教程
更多
  • 游戏攻略
  • 安卓教程
  • 苹果教程
  • 电脑教程