C#中的struct和class怎么选 C#值类型与引用类型深度解析

来源:这里教程网 时间:2026-02-21 17:34:22 作者:

struct
还是
class
,核心看两点:数据是否小而简单、是否需要值语义。小、不可变、逻辑上像基本类型(比如坐标、颜色、金额)——优先用
struct
;其余绝大多数场景,用
class
更安全、更灵活。

struct 适合这些情况

它本质是值类型,分配在栈上(小对象时),复制是整块拷贝,语义上“等于内容相同”。适合:

数据量小(一般建议 ≤ 16 字节,如
Point
DateTime
Guid
逻辑上不可变(推荐只读字段 + init-only 属性,避免意外修改) 不需要继承、多态或复杂生命周期管理 高频创建/销毁且对性能敏感(如数学计算中的向量、矩阵)

⚠️ 注意:如果 struct 成员包含引用类型(如

string
或集合),它仍是值类型,但拷贝时只复制引用,底层对象仍共享——这不是“深拷贝”,别误以为安全隔离了所有数据。

class 是默认且更稳妥的选择

它是引用类型,分配在堆上,变量存的是引用。天然支持继承、虚方法、接口实现、析构逻辑等。适合:

数据较复杂或体积较大(避免栈溢出和无谓拷贝) 需要多态行为(如不同形状的
Draw()
实现)
对象需被多个地方共享并协同修改(如订单、用户会话) 需要懒加载、事件、资源管理(
IDisposable
)等高级能力

即使你定义了一个“看起来很轻”的 class,只要它未来可能扩展功能,也比后期把 struct 改成 class 安全得多——因为 struct 的值语义一旦暴露给外部,改 class 会破坏行为(比如传参从拷贝变成传引用)。

容易踩坑的细节

不是所有“小”都该用 struct。比如:

struct
赋值、传参、返回都会触发完整拷贝——如果内部有 100 个字段,性能反而差
struct
继承自
System.ValueType
,但不能显式继承其他类型,也不能作为基类被继承
装箱(boxing)会让 struct 上堆,一次装箱就失去值类型优势,还可能引发 GC 压力(如循环中把 struct 加入
List<object></object>
默认构造函数不能重写,字段必须全部初始化(C# 10+ 支持
parameterless constructor
,但仍有约束)

一个实用判断流程

写新类型前,快速问自己:

它的实例通常有多大?(字段总大小 它代表一个“值”还是一个“东西”?(
Money
是值,
BankAccount
是东西)
会不会需要子类化或依赖多态? 有没有非托管资源要释放?(struct 不支持析构函数,
IDisposable
实现也受限)
团队是否容易理解并正确使用它的值语义?(误用常源于认知偏差)

基本上就这些。不复杂,但容易忽略。

相关推荐