Record是C# 9引入的不可变引用类型,专为数据承载设计,自动实现值语义、with非破坏性修改及简洁语法,适用于DTO等场景。

Record类型是C# 9引入的不可变引用类型,专为“数据承载”场景设计,写法简洁、自动具备值语义(如Equals、GetHashCode、ToString)、支持非破坏性修改(with表达式),非常适合DTO、领域模型、配置项等场景。
基础语法:用record声明不可变数据类型
最简写法类似class,但用record关键字:
public record Person(string Name, int Age);
编译器自动生成:
- 公共只读属性
Name和Age(对应构造参数) - 基于属性值的
Equals和GetHashCode(值相等判断) - 格式化的
ToString()(如Person { Name = "张三", Age = 25 }) - 位置记录(positional record)还自带
Deconstruct方法,支持解构赋值
自定义属性与构造逻辑
如果需要私有字段、验证逻辑或额外属性,可显式定义:
public record Student(string Name, int Age)
{
public string Id { get; init; } = Guid.NewGuid().ToString();
public DateTime CreatedAt { get; init; } = DateTime.Now;
public Student(string name, int age) : this(name, age)
{
if (string.IsNullOrWhiteSpace(name))
throw new ArgumentException("Name cannot be null or empty");
}}
注意:init访问器允许在对象初始化时(如对象初始化器或with中)赋值一次,之后不可变。
用with实现非破坏性修改
record不提供setter,但可用with创建副本并修改部分属性:
var p1 = new Person("李四", 30);
var p2 = p1 with { Age = 31 }; // 新实例,Name不变,Age更新
Console.WriteLine(p1 == p2); // false(值不同)
Console.WriteLine(p1.Equals(p2)); // false
Console.WriteLine(p1 with { } == p1); // true(完全相同副本)with本质是调用编译器生成的Clone逻辑,安全高效,无需手动复制所有字段。
继承与sealed行为
record默认是sealed(不可被继承),如需继承,必须显式声明为record class并用virtual修饰:
public virtual record Animal(string Name);
public record Dog(string Name, string Breed) : Animal(Name);
子record会自动参与值比较(父类+子类所有公开属性参与Equals),也支持with链式修改。
基本上就这些。Record不是万能替代class的工具,但它让“纯数据对象”的定义变得极其干净——不用手写构造函数、属性、重载方法,也不用依赖第三方库(如AutoMapper或RecordGenerator)。只要你的类型核心诉求是“代表一组不可变的数据”,record就是更自然、更安全的选择。






