C# 记录类型(record)全面解析:从概念到最佳实践

1. 基本概念与定位

记录类型(record)是C# 9.0引入的一种新型引用类型,专门设计用于表示不可变的数据模型。它通过精简的语法提供了创建数据载体对象的便捷方式,特别适合那些以数据为中心、行为逻辑较少的场景。

2. 核心特性解析

不可变性(Immutability)

  • 记录类型的属性默认声明为只读(init-only)

  • 对象一旦创建,其状态便无法修改

  • 优势:

    • 线程安全:无需同步锁即可在并发环境中安全使用

    • 可预测性:消除由状态变更引发的副作用

    • 函数式友好:符合函数式编程的不可变数据理念

基于值的相等性(Value-based Equality)

  • 比较依据是属性值的相等性,而非对象引用

  • 编译器自动生成 Equals()GetHashCode() 等方法的实现

  • 支持 ==!= 运算符的重载

3. 语法优势与便捷特性

位置记录(Positional Records)

使用精简的声明式语法:

cs 复制代码
public record LoginUserExtensionDto(
    string UserName, 
    string Email, 
    DateTime LoginTime);

此单行声明自动生成:

  • 公共只读属性(getter)

  • 主构造函数

  • 重写的 ToString() 方法

  • 基于值的 Equals()GetHashCode() 实现

  • 解构方法(Deconstruct)

非破坏性修改(Non-destructive Mutation)

通过 with 表达式创建对象的修改副本:

cs 复制代码
var originalUser = new LoginUserExtensionDto("john", "john@email.com", DateTime.Now);
var updatedUser = originalUser with { Email = "john.new@email.com" };

4. 编译器自动生成内容详解

LoginUserExtensionDto 记录为例,编译器自动生成:

构造函数

cs 复制代码
public LoginUserExtensionDto(string UserName, string Email, DateTime LoginTime)
{
    this.UserName = UserName;
    this.Email = Email;
    this.LoginTime = LoginTime;
}

相等性比较

cs 复制代码
public virtual bool Equals(LoginUserExtensionDto other)
{
    return other != null && 
           UserName == other.UserName && 
           Email == other.Email && 
           LoginTime == other.LoginTime;
}

字符串表示

cs 复制代码
public override string ToString()
{
    return $"LoginUserExtensionDto {{ UserName = {UserName}, Email = {Email}, LoginTime = {LoginTime} }}";
}

5. 解构功能支持

记录类型天然支持解构操作:

cs 复制代码
var user = new LoginUserExtensionDto("alice", "alice@example.com", DateTime.Now);
var (name, email, time) = user; // 解构为三个变量

6. 记录类型与类的对比

特性 Record Class
默认相等性比较 基于属性值 基于对象引用
默认可变性 不可变 可变
语法简洁性 高(支持位置参数) 需要显式定义属性、方法
继承机制 支持记录类型间的继承 完整的面向对象继承
适用场景 数据载体、DTO、值对象 复杂业务逻辑、状态管理

7. 使用场景与最佳实践

推荐使用场景

  1. 数据传输对象(DTO):如API请求/响应模型、消息契约

  2. 不可变配置对象:系统配置、参数设置等

  3. 值对象:领域驱动设计中的值对象实现

  4. 函数式编程数据容器:配合模式匹配、LINQ等使用

实践建议

  • 数据优先设计:当类型的主要目的是承载数据时优先考虑记录类型

  • 线程安全需求:在并发环境中需要共享数据时,利用其不可变性优势

  • 简化相等性逻辑:当需要基于内容比较对象时,记录类型可显著减少样板代码

  • 配合模式匹配:与C#的模式匹配特性结合使用,实现更优雅的数据处理

示例:LoginUserExtensionDto 应用场景

cs 复制代码
// 作为Web API的响应模型
public record LoginUserExtensionDto(string UserName, string Email, DateTime LoginTime);

// 在控制器中使用
[HttpGet]
public LoginUserExtensionDto GetCurrentUser()
{
    return new LoginUserExtensionDto("john_doe", "john@example.com", DateTime.UtcNow);
}

// 数据比较
var user1 = new LoginUserExtensionDto("user1", "test@email.com", DateTime.Now);
var user2 = new LoginUserExtensionDto("user1", "test@email.com", DateTime.Now);

Console.WriteLine(user1 == user2); // 输出:True(基于值相等)

记录类型通过其不可变特性和基于值的语义,为C#开发者提供了一种更安全、更简洁的数据建模方式,特别适合现代应用程序开发中对数据完整性和线程安全的需求。

附录

零基础入门C#编程的绝佳选择!52节精心设计的课程,从基础语法到实战应用,手把手带你掌握C#核心技能。无论你是编程小白还是转行开发者,这套教程都能为你打下坚实的编程基础。

开发环境配置不再头疼!这里汇集了最新版本的Visual Studio安装包合集,包含社区版、专业版等多个版本,满足不同开发需求。一键下载,快速搭建你的C#开发环境。

数据库是应用开发的基石!这个SQL Server安装包合集涵盖了从2008R2到2019的多个版本,无论你是学习测试还是项目部署,都能找到合适的版本。配套教程使用,学习效果更佳!

相关推荐
缺点内向11 小时前
如何在 C# 中将 Excel 工作表拆分为多个窗格
开发语言·c#·.net·excel
yangshuquan17 小时前
关于 C# 函数参数修饰符 out 和 in 的真相
c#·参数·in·修饰符·out
全栈师17 小时前
C#中控制权限的逻辑写法
开发语言·c#
夏霞18 小时前
c# 使用vs code 创建.net8.0以及.net6.0 webApi项目的教程
开发语言·c#·.net
追逐时光者18 小时前
C#/.NET/.NET Core优秀项目和框架2025年11月简报
后端·.net
故事不长丨19 小时前
C#线程的使用
java·microsoft·c#
小小8程序员19 小时前
C# XAML中x:Type的用法详解
开发语言·ui·c#
Aevget19 小时前
界面控件DevExpress WinForms中文教程:Data Grid - 如何获取汇总值?
ui·.net·界面控件·winform·devexpress
周杰伦fans20 小时前
在C#中,`StringContent` 是 `HttpContent` 的一个派生类
开发语言·数据库·c#
苦荞米21 小时前
异步方法-C#中坑最大最深的功能
开发语言·c#