百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

提高 C# 的生产力:C# 13 更新完全指南

bigegpt 2024-08-03 11:33 6 浏览

前言

预计在 2024 年 11 月,C# 13 将与 .NET 9 一起正式发布。今年的 C# 更新主要集中在 ref struct 上进行了许多改进,并添加了许多有助于进一步提高生产力的便利功能。

本文将介绍预计将在 C# 13 中添加的功能。

注意:目前 C# 13 还未正式发布,因此以下内容可能会发生变化。

在迭代器和异步方法中使用 refref struct

在使用 C# 进行编程时,你是否经常使用 ref 变量和 Spanref struct 类型?然而,这些不能在迭代器和异步方法中使用,于是必须使用局部函数等来避免在迭代器和异步方法中直接使用 ref 变量 ref struct 类型,这非常不方便。

这个缺点在 C# 13 中得到了改善,现在迭代器和异步方法也可以使用 refref struct 了!

在迭代器中使用 refref struct 的例子:

IEnumerable<float> GetFloatNumberFromIntArray(int[] array)
{
for (int i = 0; i < array.Length; i++)
{
Span<int> span = array.AsSpan();
// 进行一些处理...
ref float v = ref Unsafe.As<int, float>(ref array[i]);
yield return v;
}
}

在异步方法中使用 ref struct 的例子:

async Task ProcessDataAsync(int[] array)
{
Span<int> span = array.AsSpan();
// 进行一些处理...
ref int element = ref span[42];
element++;
await Task.Yield();
}

为了展示功能,我使用了不适当且含糊不清的“一些处理”,不过重要的是现在可以使用 refref struct 了!

但是,有一点需要注意,ref 变量和 ref struct 类型的变量不能超出 yieldawait 的边界使用。例如,以下示例将导致编译错误。

async Task ProcessDataAsync(int[] array)
{
Span<int> span = array.AsSpan();
// 进行一些处理...
ref int element = ref span[42];
element++;
await Task.Yield();
element++; // 错误:对 element 的访问超出了 await 的边界
}

虽然我们已经说到这里,但我想可能有人会疑惑,到底 refref struct 是什么,所以我稍微解释一下。

在 C# 中,可以使用 ref 来获取变量的引用。这样,就可以通过引用来更改原始变量。以下是一个例子:

void Swap(ref int a, ref int b) // ref 表示引用
{
int temp = a;
a = b;
b = temp; // 到这里,a 和 b 已经交换了
}

int x = 1;
int y = 2;
Swap(ref x, ref y); // 获取 x 和 y 的引用,调用 Swap 来交换 x 和 y

另一方面,ref struct 是用于定义只能存在于堆栈上的值类型的。这是为了避免垃圾收集的开销。然而,由于 ref struct 只能存在于堆栈上,所以在 C# 13 之前,它不能在迭代器和异步方法等地方使用。

顺便一提,ref struct 之所以带有 ref,是因为 ref struct 的实例只能存在于堆栈上,其遵循的生命周期规则与 ref 变量相同。

allows ref struct 泛型约束

在以前,ref struct 不能作为泛型类型参数使用,因此,考虑到代码的可重用性,引入了泛型,但最终 ref struct 不能使用,必须为 SpanReadOnlySpan 重新编写相同的处理,于是就很麻烦。

在 C# 13 中,泛型类型也可以使用 ref struct 了:

using System;
using System.Numerics;

Process([1, 2, 3, 4], Sum); // 10
Process([1, 2, 3, 4], Multiply); // 24

T Process<T>(ReadOnlySpan<T> span, Func<ReadOnlySpan<T>, T> method)
{
return method(span);
}

T Sum<T>(ReadOnlySpan<T> span) where T : INumberBase<T>
{
T result = T.Zero;
foreach (T value in span)
{
result += value;
}
return result;
}

T Multiply<T>(ReadOnlySpan<T> span) where T : INumberBase<T>
{
T result = T.One;
foreach (T value in span)
{
result *= value;
}
return result;
}

为什么像 ReadOnlySpan<T> 这样的 ref struct 类型可以作为 Func 的类型参数呢?为了调查这个问题,我查看了 .NET 的 源代码,发现 Func 类型的泛型参数是这样定义的:

public delegate TResult Func<in T, out TResult>(T arg)
where T : allows ref struct
where TResult : allows ref struct;

如果在泛型参数上添加 allow ref struct 约束,那么就可以将 ref struct 类型传递给该参数。

这确实是一个方便的功能。

ref struct 也可以实现接口

在 C# 13 中,ref struct 可以实现接口。

如果将此功能与 allows ref struct 结合使用,那么也可以通过泛型类型传递引用:

using System;
using System.Numerics;

int a = 10;
// 使用 Ref<int> 保存 a 的引用
Ref<int> aRef = new Ref<int>(ref a);
// 传递 Ref<int>
Increase<Ref<int>, int>(aRef);
Console.WriteLine(a); // 11

void Increase<T, U>(T data) where T : IRef<U>, allows ref struct where U : INumberBase<U>
{
ref U value = ref data.GetRef();
value++;
}

interface IRef<T>
{
ref T GetRef();
}

// 为 Ref<T> 这样的 ref struct 实现接口
ref struct Ref<T> : IRef<T>
{
private ref T _value;

public Ref(ref T value)
{
_value = ref value;
}

public ref T GetRef()
{
return ref _value;
}
}

这样一来,编写 ref struct 相关的代码就变得更容易了。另外,也能给各种 ref struct 实现的枚举器实现 IEnumerator 之类的接口了。

集合类型和 Span 也可以使用 params

在以前,params 只能用于数组类型,但从 C# 13 开始,它也可以用于其他集合类型和 Span

params 是一种功能,允许在调用方法时直接指定任意数量的参数。

例如,

Test(1, 2, 3, 4, 5, 6);
void Test(params int[] values) { }

如上所示,可以直接指定任意数量的 int 参数。

从 C# 13 开始,除了数组类型外,其他集合类型、SpanReadOnlySpan 类型以及与集合相关的接口也可以添加 params

Test(1, 2, 3, 4, 5, 6);
void Test(params ReadOnlySpan<int> values) { }

// 或者
Test(1, 2, 3, 4, 5, 6);
void Test(params List<int> values) { }

// 接口也可以
Test(1, 2, 3, 4, 5, 6);
void Test(params IEnumerable<int> values) { }

这也很方便!

field 关键字

在实现 C# 的属性时,经常需要定义一大堆字段,如下所示...

partial class ViewModel : INotifyPropertyChanged
{
// 定义字段
private int _myProperty;

public int MyProperty
{
get => _myProperty;
set
{
if (_myProperty != value)
{
_myProperty = value;
OnPropertyChanged();
}
}
}
}

因此,从 C# 13 开始,field 关键字将派上用场!

partial class ViewModel : INotifyPropertyChanged
{
public int MyProperty
{
// 只需使用 field
get => field;
set
{
if (field != value)
{
field = value;
OnPropertyChanged();
}
}
}
}

不再需要自己定义字段,只需使用 field 关键字,字段就会自动生成。

这也非常方便!

部分属性

在编写 C# 时,常见的问题之一是:属性不能添加 partial 修饰符。

在 C# 中,可以在类或方法上添加 partial,以便分别进行声明和实现。此外,还可以分散类的各个部分。它的主要用途是在使用源代码生成器等自动生成工具时,指定要生成的内容。

例如:

partial class ViewModel
{
// 这里只声明方法,实现部分由工具自动生成
partial void OnPropertyChanged(string propertyName);
}

然后自动生成工具会生成以下代码:

partial class ViewModel : INotifyPropertyChanged
{
public event PropertyChangedEventHandler? PropertyChanged;

partial void OnPropertyChanged(string propertyName)
{
PropertyChanged?.Invoke(this, new(propertyName));
}
}

开发者只需要声明 OnPropertyChanged,其实现将全部由自动生成,从而节省了开发者的时间。

从 C# 13 开始,属性也支持 partial

partial class ViewModel
{
// 声明部分属性
public partial int MyProperty { get; set; }
}

partial class ViewModel
{
// 部分属性的实现
public partial int MyProperty
{
get
{
// ...
}
set
{
// ...
}
}
}

这样,属性也可以由工具自动生成了。

锁对象

众所周知,lock 是一种功能,通过监视器用于线程同步。

object lockObject = new object();
lock (lockObject)
{
// 关键区
}

但是,这个功能的开销其实很大,会影响性能。

为了解决这个问题,C# 13 实现了锁对象。要使用此功能,只需用 System.Threading.Lock 替换被锁定的对象即可:

using System.Threading;

Lock lockObject = new Lock();
lock (lockObject)
{
// 关键区
}

这样就可以轻松提高性能了。

初始化器中的尾部索引

索引运算符 ^ 可用于表示集合末尾的相对位置。从 C# 13 开始,初始化器也支持此功能:

var x = new Numbers
{
Values =
{
[1] = 111,
[^1] = 999 // ^1 是从末尾开始的第一个元素
}
// x.Values[1] 是 111
// x.Values[9] 是 999,因为 Values[9] 是最后一个元素
};

class Numbers
{
public int[] Values { get; set; } = new int[10];
}

ESCAPE 字符

在 Unicode 字符串中,可以使用 \e 代替 \u001b\x1b\u001b\x1b\e 都表示 ESCAPE 字符。它们通常用于表示控制字符。

  • \u001b 表示 Unicode 转义序列,\u 后面的 4 位十六进制数表示 Unicode 代码点

  • \x1b 表示十六进制转义序列,\x 后面的 2 位十六进制数表示 ASCII 代码

  • \e 表示 ESCAPE 字符本身

推荐使用 \e 的原因是,可以避免在十六进制中的混淆。

例如,如果 \x1b 后面跟着 3,则变为 \x1b3,由于 \x1b3 之间没有明确的分隔,因此不清楚应该分别解释成 \x1b3,还是放在一起解释。

如果使用 \e,则可以避免混淆。

其他

结语

C# 正在年复一年地进化,对我来说 C# 13 的更新中实现了许多非常实用且方便的功能,解决了不少实际的痛点。期待 .NET 9 和 C# 13 的正式发布~

相关推荐

【机器学习】数据挖掘神器LightGBM详解(附代码)

来源:机器学习初学者本文约11000字,建议阅读20分钟本文为你介绍数据挖掘神器LightGBM。LightGBM是微软开发的boosting集成模型,和XGBoost一样是对GBDT...

3分钟,用DeepSeek全自动生成语音计算器,还带括号表达式!

最近,大家慢慢了解到了DeepSeek的强大功能,特别是它在编程领域也同样强大。编程零基础小白,一行代码不用写,也能全自动生成一个完整的、可运行的软件来!很多程序员一直不相信小白不写代码也能编软件!下...

python学习笔记 3.表达式

在Python中,表达式是由值、变量和运算符组成的组合。以下是一些常见的Python表达式:算术表达式:由数值和算术运算符组成的表达式,如加减乘除等。例如:5+3、7*2、10/3等。字符...

5.7 VS 8.x,为什么用户不升级MySql

一般来说为了更好的功能和性能,都需要将软件升级到最新的版本,然而在开源软件中,由于一些开发商变化或其他的问题(开源授权变化),致使人们不愿使用最新的版本,一个最典型的问题就是CentOS操作系统。还有...

大厂高频:讲一下MySQL主从复制

大家经常听说主从复制,那么主从复制的意义?能解决的问题有哪些?主从复制能解决的问题就是在我们平时开发的程序中操作数据库的时候,大多数的情况查询的操作大大超过了写的操作,也就说对数据库读取数据的压力比较...

MYSQL数据库的五大安全防护措施

以技术为基础的企业里最有价值的资产莫过于是客户或者其数据库中的产品信息了。因此,在这样的企业中,保证数据库免受外界攻击是数据库管理的重要环节。很多数据库管理员并没有实施什么数据库保护措施,只是因为觉得...

docker安装mysql

准备工作已安装Docker环境(官方安装文档)终端/命令行工具(Linux/macOS/WSL)步骤1:拉取MySQL镜像打开终端执行以下命令,拉取官方MySQL镜像(默认最新版本):d...

Zabbix监控系统系列之六:监控 mysql

zabbix监控mysql1、监控规划在创建监控项之前要尽量考虑清楚要监控什么,怎么监控,监控数据如何存储,监控数据如何展现,如何处理报警等。要进行监控的系统规划需要对Zabbix很了解,这里只是...

详解MySQL的配置文件及优化

#头条创作挑战赛#在Windows系统中,MySQL服务器启动时最先读取的是my.ini这个配置文件。在Linux系统中,配置文件为my.cnf,其路径一般为/etc/my.cnf或/etc/mysq...

Mysql 几个批处理执行脚本

学习mysql过程中,需要创建测试数据,并让多人每人一个数据库连接并进行作业检查。整合部分批处理创建数据批量创建数据库DELIMITER$CREATEPROCEDURECreateDatab...

MySQL学到什么程度?才有可以在简历上写精通

前言如今互联网行业用的最多就是MySQL,然而对于高级Web面试者,尤其对于寻找30k下工作的求职者,很多MySQL相关知识点基本都会涉及,如果面试中,你的相关知识答的模糊和不切要点,基...

mysql 主、从服务器配置“Slave_IO_Running: Connecting” 问题分析

#在进行mysql主、从服务器配置时,”SHOWSLAVESTATUS;“查看从库状态Slave_IO_Runing,出现错误:“Slave_IO_Running:Connectin...

MYSQL数据同步

java开发工程师在实际的开发经常会需要实现两台不同机器上的MySQL数据库的数据同步,要解决这个问题不难,无非就是mysql数据库的数据同步问题。但要看你是一次性的数据同步需求,还是定时数据同步,亦...

「MySQL 8」MySQL 5.7都即将停只维护了,是时候学习一波MySQL 8了

MySQL8新特性选择MySQL8的背景:MySQL5.6已经停止版本更新了,对于MySQL5.7版本,其将于2023年10月31日停止支持。后续官方将不再进行后续的代码维护。另外,...

Prometheus监控mysql

通过Prometheus监控Mysql,我们需要在Mysql端安装一个mysql-exporter,然后Prometheus通过mysql-exporter暴露的端口抓取数据。1.安装一个MYSQL配...