【.NET Core】记录(Record)详解

简介: 【.NET Core】记录(Record)详解

一、什么是记录

record修饰符定义一个引用类型,用来提供用于封装数据的内置功能。C# 10允许recode class语法作为同义词来阐明引用类型,并允许recode struct使用相同功能定义值类型。本文将重点讲解recode在实际中的使用。


在记录(record)上声明主构造函数时,编译器会为主构造函数参数生成公共属性。记录的主构造函数参数称为位置参数。编译器创建镜像主构造函数或位置参数的位置属性。编译器不会在没有record修饰符的类型上合成主构造函数参数的属性。


二、定义一个记录

2.1 使用record(或record class)定义引用类型

public record Person(string name,string address,int age);
public record Person
{
    public required string name{get;init;}
    public required string address{get;init;}
    public required int address{get;init;}
}

2.2 使用record struct定义值类型

public readonly record struct Point(double X,double Y,double Z);
public record struct Point
{
    public double X{get;init;}
    public double Y{get;init;}
    public double Z{get;init;}
}

记录(record)定义了一些不可变的类型,其实记录也可以创建为可变的类型。但是record主要用于支持不可变的数据模型中。记录创建为可变属性和字段的记录如下:

2.3 创建可变记录类型

创建可变引用类型记录

public record Person
{
    public required string FirstName { get; set; }
    public required string LastName { get; set; }
};

创建结构型记录

public record struct Point
{
    public double X { get; set; }
    public double Y { get; set; }
    public double Z { get; set; }
}

三、记录类型提供哪些功能

  • 用于创建具有不可变属性的引用类型的简明语法。
  • 内置行为对于数据为中心的引用类型可用于:
  • 值相等性
  • 非破坏性变化的简明语法
  • 用于显示的内置格式设置
  • 支持继承层次结构


四、引用类型记录和值类型记录区别

  • recordrecord class声明引用类型。class关键字是可选项,但可以提供代码辨识度。record struct声明值类型。
  • 记录属性和readonly record struct中不可变。

record一词用于描述应用于所有记录类型的行为。record structrecord class用于描述仅适用于struct或class类型的行为。


五、Recode常用属性

5.1 属性定义的位置语法

在创建Recode实例时,可以使用位置参数来声明记录的属性,并初始化属性值:

public record Person(string UserName,int UserAge);
public static void Main()
{
    Person person=new("诸葛亮",59);
    Console.WriteLine(person);
}

属性定义使用位置语法时,编译器将执行一下操作:


  • 为记录声明中提供的每个位置参数提供一个公共的自动实现的属性。
  • 对于record和readonly record struct类型,为record属性。
  • 对于record struct类型,为读写属性。
  • 主构造函数,它的参数与记录声明上的位置参数匹配。
  • 对于record struct类型,则是将每个字段设置为其默认值的无参数构造函数。
  • 一个Deconstruct方法,对记录声明中提供的每个位置参数都一个out参数。此方法解构了使用位置语法定义的属性;并忽略了使用标准的属性语法定义的属性。

如果生成的自动实现的属性定义并不是你所需要的,你可以自行定义同名的属性。例如,你可能想要更改可访问性或可变性,或者为get 或set访问器提供实现。如果在源中声明属性,则必须从记录的位置参数初始化该属性。如果属性是自动实现的属性,则必须初始化该属性。如果在源中添加支持字段,则必须初始化支持字段。生成的析构函数将使用属性定义。


5.2 不可变性

positional record和positional readonly record struct声明init-only属性。positional record struct声明read-write属性性,可以替代这些默认值中的任何一个。


如果你需要一个以数据为中心的类型是线程安全的,或者需要使哈希代码保持不变,那么不可变性很有用。但不可变性并不是适用于所有的数据场景。例如,Entity Framework Core就不支持通过不可变实体类型进行更新。


init-only属性无论是通过位置参数(record class和readonly record struct)创建的,还是通过指定init访问器创建的,都具有浅的不可变性。初始化后,将不能更改值型的值或引用类型的引用地址。不过,引用类型引用的数据是可以更改的,下面示例展示了引用型不可变属性的内容时可变的:

public record Person(string UserName,int UserAge,String[] UserPhones);
public static void Main()
{
    Person person=new("诸葛亮",55,new string[1]{"010-6689012"});
    Console.WriteLine(person.UserPhones[0]);// output:010-6689012
    person.PhoneNumbers[0] = "010-6689018";
    Console.WriteLine(person.UserPhones[0]);// output:010-6689018  
}

记录类型持有功能是由编译器合成的方法实现的,这些方法都不会通过修改对象状态来影响不可变性。除非另行指定,否则将为record,record structreadonly record struct声明生成综合方法。

5.3 值相等性

如果不替代或替换相等性方法,则声明的类型将控制如何定义相等性。

  • 对于class类型,如果两个对象引用内存中的同一个对象,则这两个对象相等。
  • 对于struct类型,如果两个对象是相同的类型并存储相同的值,则这两个对象相等。
  • 对于具有record修饰符(record class、record struct和readonly record struct)的类型,如果两个对象是相同的类型并存储相同的值,则这两个对象相等。

record struct的相等性定义与struct的相等性定义相同。不同之处在于,对于struct,实现处于ValueType.Equals(Object)中并且依赖反射,对于记录,实现由编译器合成,并且使用声明的数据成员。

5.4 非破坏性变化

如果需要复制包含一些修饰的实例,可以使用with表达式来实现非破坏性变化。with表达式创建一个新的记录实例,该实例是现有记录的实例的一个副本,修改了指定属性和字段。使用对象初始化设定项语法指定要更改的值,实例如下:

public record Person(string userName,int userAge)
{
   public string[] userPhone {get;init;}
}
public static void Main()
{
  Person person1=new("诸葛亮",35);
  Console.WriteLine(person1);
  //output:Person{userName=诸葛亮,userAge=35,userPhone=System.String[]}
  Person person2= person1 with {userName="赵云"};
  Console.WriteLine(person2);
  //output:Person{userName=赵云,userAge=35,userPhone=System.String[]}
  Console.WriteLine(person1 == person2);
  //output:False
    
}

with表达式可以设置位置属性或使用标准属性语法创建的属性。显示声明属性必须有一个init或set访问器才能在with表达式中进行更改。


with表达式的结果是一个浅的副本,这意味着对于引用属性,只复制对实例的引用。原始记录和副本最终都具有对同一个实例的引用。


为了对record class类型实现此功能,编译器合成了一个克隆方法和一个复制构造函数,虚拟克隆方法返回由复制构造函数初始化的新记录,当使用with表达式时,编译器将创建调用克隆方法的代码,然后设置with表达式中指定的属性。


5.5 继承

record继承仅适用于record class类型。一条记录可以从另一条记录继承。但是记录不能继承类,类也不能继承记录类。


派生记录为基本记录主构造函数中的所有参数声明位置参数。基本记录声明并初始化这些属性。派生记录不会隐藏它们,而只会创建和初始化在基本记录中声明的参数的属性。

public abstract record Person(string FirstName, string LastName);
public record Teacher(string FirstName, string LastName, int Grade)
    : Person(FirstName, LastName);
public static void Main()
{
    Person teacher = new Teacher("Nancy", "Davolio", 3);
    Console.WriteLine(teacher);
    // output: Teacher { FirstName = Nancy, LastName = Davolio, Grade = 3 }
}

六、总结

记录是一个语法糖,本质上还是class或者struct,它只编译时生效,在公共语言运行时中并不存在记录这个内容。

目录
相关文章
|
25天前
|
开发框架 .NET 开发者
简化 ASP.NET Core 依赖注入(DI)注册-Scrutor
Scrutor 是一个简化 ASP.NET Core 应用程序中依赖注入(DI)注册过程的开源库,支持自动扫描和注册服务。通过简单的配置,开发者可以轻松地从指定程序集中筛选、注册服务,并设置其生命周期,同时支持服务装饰等高级功能。适用于大型项目,提高代码的可维护性和简洁性。仓库地址:<https://github.com/khellang/Scrutor>
40 5
|
3月前
|
存储 开发框架 JSON
ASP.NET Core OData 9 正式发布
【10月更文挑战第8天】Microsoft 在 2024 年 8 月 30 日宣布推出 ASP.NET Core OData 9,此版本与 .NET 8 的 OData 库保持一致,改进了数据编码以符合 OData 规范,并放弃了对旧版 .NET Framework 的支持,仅支持 .NET 8 及更高版本。新版本引入了更快的 JSON 编写器 `System.Text.UTF8JsonWriter`,优化了内存使用和序列化速度。
101 0
|
2月前
|
开发框架 .NET C#
在 ASP.NET Core 中创建 gRPC 客户端和服务器
本文介绍了如何使用 gRPC 框架搭建一个简单的“Hello World”示例。首先创建了一个名为 GrpcDemo 的解决方案,其中包含一个 gRPC 服务端项目 GrpcServer 和一个客户端项目 GrpcClient。服务端通过定义 `greeter.proto` 文件中的服务和消息类型,实现了一个简单的问候服务 `GreeterService`。客户端则通过 gRPC 客户端库连接到服务端并调用其 `SayHello` 方法,展示了 gRPC 在 C# 中的基本使用方法。
46 5
在 ASP.NET Core 中创建 gRPC 客户端和服务器
|
1月前
|
开发框架 缓存 .NET
GraphQL 与 ASP.NET Core 集成:从入门到精通
本文详细介绍了如何在ASP.NET Core中集成GraphQL,包括安装必要的NuGet包、创建GraphQL Schema、配置GraphQL服务等步骤。同时,文章还探讨了常见问题及其解决方法,如处理复杂查询、错误处理、性能优化和实现认证授权等,旨在帮助开发者构建灵活且高效的API。
28 3
|
9天前
|
开发框架 算法 中间件
ASP.NET Core 中的速率限制中间件
在ASP.NET Core中,速率限制中间件用于控制客户端请求速率,防止服务器过载并提高安全性。通过`AddRateLimiter`注册服务,并配置不同策略如固定窗口、滑动窗口、令牌桶和并发限制。这些策略可在全局、控制器或动作级别应用,支持自定义响应处理。使用中间件`UseRateLimiter`启用限流功能,并可通过属性禁用特定控制器或动作的限流。这有助于有效保护API免受滥用和过载。 欢迎关注我的公众号:Net分享 (239字符)
25 0
|
4月前
|
开发框架 监控 前端开发
在 ASP.NET Core Web API 中使用操作筛选器统一处理通用操作
【9月更文挑战第27天】操作筛选器是ASP.NET Core MVC和Web API中的一种过滤器,可在操作方法执行前后运行代码,适用于日志记录、性能监控和验证等场景。通过实现`IActionFilter`接口的`OnActionExecuting`和`OnActionExecuted`方法,可以统一处理日志、验证及异常。创建并注册自定义筛选器类,能提升代码的可维护性和复用性。
|
4月前
|
开发框架 .NET 中间件
ASP.NET Core Web 开发浅谈
本文介绍ASP.NET Core,一个轻量级、开源的跨平台框架,专为构建高性能Web应用设计。通过简单步骤,你将学会创建首个Web应用。文章还深入探讨了路由配置、依赖注入及安全性配置等常见问题,并提供了实用示例代码以助于理解与避免错误,帮助开发者更好地掌握ASP.NET Core的核心概念。
118 3
|
3月前
|
开发框架 JavaScript 前端开发
一个适用于 ASP.NET Core 的轻量级插件框架
一个适用于 ASP.NET Core 的轻量级插件框架
|
4月前
|
开发框架 NoSQL .NET
利用分布式锁在ASP.NET Core中实现防抖
【9月更文挑战第5天】在 ASP.NET Core 中,可通过分布式锁实现防抖功能,仅处理连续相同请求中的首个请求,其余请求返回 204 No Content,直至锁释放。具体步骤包括:安装分布式锁库如 `StackExchange.Redis`;创建分布式锁服务接口及其实现;构建防抖中间件;并在 `Startup.cs` 中注册相关服务和中间件。这一机制有效避免了短时间内重复操作的问题。
|
5月前
|
开发框架 前端开发 中间件
聊聊 ASP.NET Core 中间件(二):中间件和筛选器的区别
聊聊 ASP.NET Core 中间件(二):中间件和筛选器的区别
170 1