0

0

C# 创建自定义异常方法 C#如何定义和抛出自定义异常

煙雲

煙雲

发布时间:2026-02-26 11:26:57

|

188人浏览过

|

来源于php中文网

原创

自定义异常类必须继承exception或其子类,需提供三个标准构造函数并添加[serializable]特性;抛出时应传递innerexception,重抛用throw而非throw ex;异常仅用于意外情况,非控制流手段。

c# 创建自定义异常方法 c#如何定义和抛出自定义异常

自定义异常类必须继承 Exception 或其子类

不继承 Exception 的类无法被 catch 捕获为异常,编译器会报错。常见错误是直接写一个普通类,然后用 throw new MyException() —— 这会导致 CS0155(无法捕获非异常类型)。

推荐继承 Exception,而非 System.ApplicationException(已过时,文档明确不建议使用)。

  • 必须提供至少三个构造函数:无参、string messagestring message, Exception innerException
  • 若需序列化(如跨 AppDomain 或 WCF 场景),需添加 [Serializable] 特性
  • 避免在构造函数中做耗时操作或依赖外部状态
[Serializable]
public class InsufficientBalanceException : Exception
{
    public decimal CurrentBalance { get; }
    public decimal RequiredAmount { get; }

    public InsufficientBalanceException() : base("账户余额不足") { }

    public InsufficientBalanceException(string message) : base(message) { }

    public InsufficientBalanceException(string message, decimal current, decimal required)
        : base(message)
    {
        CurrentBalance = current;
        RequiredAmount = required;
    }

    public InsufficientBalanceException(string message, Exception innerException)
        : base(message, innerException) { }

    protected InsufficientBalanceException(SerializationInfo info, StreamingContext context)
        : base(info, context)
    {
        CurrentBalance = info.GetDecimal("CurrentBalance");
        RequiredAmount = info.GetDecimal("RequiredAmount");
    }

    public override void GetObjectData(SerializationInfo info, StreamingContext context)
    {
        base.GetObjectData(info, context);
        info.AddValue("CurrentBalance", CurrentBalance);
        info.AddValue("RequiredAmount", RequiredAmount);
    }
}

抛出时优先用带业务语义的构造函数

不要只用无参构造,丢失关键上下文。调用方靠 ex.Message 做日志或用户提示时,信息越具体越好;靠属性做程序逻辑分支时(比如重试策略),属性必须被正确初始化。

Descript
Descript

一个多功能的音频和视频编辑引擎

下载
  • 简单场景:用 new InsufficientBalanceException($"余额{cur}不足以支付{req}")
  • 需要结构化数据参与后续处理:用带参数的构造函数,确保 CurrentBalanceRequiredAmount 可靠赋值
  • 包装底层异常时,务必传入 innerException,否则堆栈和原始错误信息丢失

throwthrow ex 的区别直接影响调试体验

catch 块中重新抛出异常时,写 throw ex; 会清空原始堆栈,只剩当前行;而裸 throw; 保留完整原始调用链。这是线上排查最常踩的坑之一。

  • 错误写法:catch (SqlException ex) { throw ex; } → 堆栈从这里截断
  • 正确写法:catch (SqlException ex) { throw new DataAccessException("数据库操作失败", ex); }(带 inner)
  • 或仅做日志后透传:catch (Exception ex) { Log(ex); throw; }(注意不是 throw ex;

自定义异常不应替代返回值或状态码

异常用于“意外”情况,不是控制流手段。比如验证失败(用户名为空)、参数校验不通过,更适合返回 Result<t></t>ValidationResult,而不是抛 InvalidInputException

  • 适合抛异常:文件被占用、网络超时、数据库连接中断、权限突然失效
  • 不适合抛异常:用户输入邮箱格式错误、订单状态非法(应走状态机校验)
  • 过度使用自定义异常会导致 try/catch 泛滥,掩盖真正需要关注的故障点
实际项目里最容易被忽略的是序列化构造函数和 GetObjectData 的配对实现——缺一不可,否则在 .NET Core+ 跨进程场景下反序列化会失败,且错误提示极不友好。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

850

2023.08.02

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

850

2023.08.02

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

425

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

597

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

425

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

597

2023.08.10

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

377

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2095

2023.08.14

batoto漫画官网入口与网页版访问指南
batoto漫画官网入口与网页版访问指南

本专题系统整理batoto漫画官方网站最新可用入口,涵盖最新官网地址、网页版登录页面及防走失访问方式说明,帮助用户快速找到batoto漫画官方平台,稳定在线阅读各类漫画内容。

331

2026.02.25

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10.2万人学习

C 教程
C 教程

共75课时 | 5万人学习

C++教程
C++教程

共115课时 | 19.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号