找回密码
 立即注册
首页 业界区 业界 .NET 10 新增功能系列文章5——C# 14 中的新增功能 ...

.NET 10 新增功能系列文章5——C# 14 中的新增功能

扈怀易 2025-8-7 11:13:25
1. 扩展成员:更强大的扩展能力

C# 14对扩展方法进行了重大增强,引入了全新的扩展成员语法。这一新特性不仅支持扩展方法,还允许开发者声明扩展属性静态扩展成员
  1. public static class Enumerable
  2. {
  3.     // 实例扩展成员块
  4.     extension<TSource>(IEnumerable<TSource> source)
  5.     {
  6.         // 扩展属性
  7.         public bool IsEmpty => !source.Any();
  8.         
  9.         // 扩展索引器
  10.         public TSource this[int index] => source.Skip(index).First();
  11.         
  12.         // 扩展方法
  13.         public IEnumerable<TSource> Where(Func<TSource, bool> predicate) { ... }
  14.     }
  15.     // 静态扩展成员块
  16.     extension<TSource>(IEnumerable<TSource>)
  17.     {
  18.         // 静态扩展方法
  19.         public static IEnumerable<TSource> Combine(IEnumerable<TSource> first, IEnumerable<TSource> second) { ... }
  20.         
  21.         // 静态扩展属性
  22.         public static IEnumerable<TSource> Identity => Enumerable.Empty<TSource>();
  23.     }
  24. }
  25. 实例扩展成员可以像普通实例成员一样调用,如`sequence.IsEmpty`,而静态扩展成员则通过类型名调用,如`IEnumerable<int>.Identity`。这一特性极大地丰富了扩展方法的应用场景,使API设计更加灵活。
复制代码
2. field关键字:简化属性实现

C# 14引入了field上下文关键字,用于简化属性实现,无需显式声明后备字段。编译器会自动合成支持字段。
改进前的传统实现:
  1. private string _msg;
  2. public string Message
  3. {
  4.     get => _msg;
  5.     set => _msg = value ?? throw new ArgumentNullException(nameof(value));
  6. }
复制代码
使用field关键字后的简化实现:
  1. public string Message
  2. {
  3.     get;
  4.     set => field = value ?? throw new ArgumentNullException(nameof(value));
  5. }
复制代码
field关键字使属性实现更加简洁,特别是在需要对属性值进行验证或转换时。如果类型中已有名为field的符号,可以使用@field或this.field来消除歧义。
3. 隐式Span转换:提升性能关键代码

C# 14增强了对System.Span和System.ReadOnlySpan的支持,引入了更多隐式转换,使这些高性能类型的使用更加自然。
新特性包括:

  • T[]到Span的隐式转换
  • Span到ReadOnlySpan的隐式转换
  • string到ReadOnlySpan的隐式转换
这些转换使得Span类型可以:

  • 作为扩展方法的接收器
  • 与其他转换组合使用
  • 在泛型类型推断场景中提供帮助
Span类型的这些改进对于高性能场景(如文本处理、数值计算等)尤为重要,可以在不牺牲安全性的前提下获得接近原生代码的性能。
4. nameof支持未绑定泛型类型

C# 14扩展了nameof操作符的功能,使其支持未绑定的泛型类型。这使得在反射、日志记录等场景下的代码更加简洁。
  1. // C# 14新特性
  2. string name = nameof(List<>); // 返回"List"
  3. // 之前只能这样做
  4. string name = nameof(List<int>); // 返回"List"
复制代码
这一改进特别适用于需要处理泛型类型名称的通用框架代码,减少了硬编码字符串的使用,提高了代码的维护性。
5. 带修饰符的简单Lambda参数

C# 14允许在不指定参数类型的情况下,为lambda表达式参数添加修饰符(如ref、in、out等),进一步简化了lambda表达式的书写。
改进前:
  1. TryParse<int> parse2 = (string text, out int result) => Int32.TryParse(text, out result);
复制代码
改进后:
  1. TryParse<int> parse1 = (text, out result) => Int32.TryParse(text, out result);
复制代码
注意params修饰符仍然需要显式类型声明。这一特性使lambda表达式更加简洁,特别是在处理包含out参数的方法时。
6. 部分成员增强:构造函数和事件

C# 14扩展了部分成员的支持范围,现在可以将实例构造函数事件声明为部分成员。
部分构造函数的要点:

  • 必须包含定义声明和实现声明
  • 只有实现声明可以包含构造函数初始值设定项(this()或base())
  • 只有一个分部类型声明可以包含主构造函数语法
部分事件的要点:

  • 必须包含定义声明和实现声明
  • 实现声明必须包含add和remove访问器
  • 定义声明类似于字段事件
这些增强使得代码生成器和手动编写代码的协作更加灵活,特别适合大型项目或框架开发。
7. 用户自定义复合赋值运算符

C# 14引入了用户自定义复合赋值运算符的能力,允许开发者为自己定义的类型重载复合赋值运算符(如+=、-=等)。这一特性通过更自然的语法简化了对自定义类型的操作。
虽然具体的实现细节需要参考功能规范,但这一特性为数值计算库、矩阵运算等场景提供了更优雅的语法支持。
8. 空条件赋值:更简洁的null检查

C# 14扩展了空条件运算符?.的用法,现在可以将其放在赋值操作的左侧,实现空条件赋值。
传统null检查方式:
  1. if (customer is not null)
  2. {
  3.     customer.Order = GetCurrentOrder();
  4. }
复制代码
使用空条件赋值的简洁方式:
  1. customer?.Order = GetCurrentOrder();
复制代码
在这个例子中,GetCurrentOrder()方法仅在customer不为null时才会被调用。这一特性同样适用于复合赋值运算符(如+=、-=等),但不支持递增(++)和递减(--)运算符。
系列文章

.NET 10 中的新增功能系列文章1——运行时中的新增功能
.NET 10 中的新增功能系列文章2——ASP.NET Core 中的新增功能
.NET 10 中的新增功能系列文章3——.NET MAUI 中的新增功能
.NET 10 中的新增功能系列文章4——.NET SDK中的新增功能

来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
您需要登录后才可以回帖 登录 | 立即注册