C# 13 中的新增功能实操

news/2025/3/31 23:30:09/文章来源:https://www.cnblogs.com/Can-daydayup/p/18798838

前言

今天大姚带领大家一起来看看 C# 13 中的新增几大功能,并了解其功能特性和实际应用场景。

前提准备

要体验 C# 13 新增的功能可以使用最新的 Visual Studio 2022 版本或 .NET 9 SDK 尝试这些功能。

Visual Studio 2022安装

  • https://visualstudio.microsoft.com/zh-hans/downloads

.NET 9 SDK

  • https://dotnet.microsoft.com/zh-cn/download/dotnet/9.0

 

params 集合

在 C# 13 中,params 的改进使其从可变数量的数组参数升级为可变数量的集合类型参数。这一改进通过支持高性能集合类型(如Span<T>、ReadOnlySpan<T>)和简化调用语法,显著提升了代码的灵活性和效率。

C# 13 中的 params 集合变得类型更灵活,满足更复杂的应用场景,并且性能得到了进一步的提升。

在 C# 13 之前:

params 仅支持一维数组(如params int[] listparams object[] list)。调用方法时需显式传递数组或数组元素类型的参数的逗号分隔列表。

在 C# 13 中:

params 修饰符并不局限于数组类型。 现在可以将 params 用于任何已识别的集合类型,包括 System.Span<T>、System.ReadOnlySpan<T>,以及那些实现 System.Collections.Generic.IEnumerable<T> 并具有 Add 方法的类型。 除了具体类型外,还可以使用接口 System.Collections.Generic.IEnumerable<T>、System.Collections.Generic.IReadOnlyCollection<T>、System.Collections.Generic.IReadOnlyList<T>、System.Collections.Generic.ICollection<T>和 System.Collections.Generic.IList<T>

        public static void SpanDataPrintRun()
        {
            Span<int> originalSpan = [1, 2, 3, 4, 5];
            SpanDataPrint(originalSpan);
        }

        public static void SpanDataPrint<T>(params Span<T> spans)
        {
            for (int i = 0; i < spans.Length; i++)
            {
                Console.WriteLine(spans[i]);
            }
        }

新增Lock锁对象

.NET 9 包含一种新的用于互斥的 System.Threading.Lock 类型,比仅在任意 System.Object 实例上进行锁定更有效。该类型通过其 API 提供更好的线程同步,通过Lock.EnterScope()返回的ref struct自动管理锁的释放,减少死锁风险。

  • System.Threading.Lock类型提案:https://github.com/dotnet/runtime/issues/34812

        private object _oldLock = new object();
        private System.Threading.Lock _newLock = new System.Threading.Lock();

        public void LockTest()
        {
            lock (_oldLock)
            {
                Console.WriteLine("Old lock");
            }


            lock (_newLock)
            {
                // 传统 lock 语法(优化版)
            }

            using (_newLock.EnterScope())
            {
                //  作用域自动释放(推荐写法)
            }

            _newLock.Enter();
            try
            {
                // 显式 Enter/Exit 调用
            }
            finally { _newLock.Exit(); }

            if (_newLock.TryEnter())
            {
                try
                {
                    // 非阻塞尝试获取锁
                }
                finally { _newLock.Exit(); }
            }

        }

新的转义序列

新增 \e 转义字符作为 ESCAPE 字符 Unicode U+001B 的字符文本转义序列。以前,只能使用的是 \u001b 或 \x1b。不建议使用\x1b,因为如果 1b 后面的下一个字符是有效的十六进制数字,则那些字符会成为转义序列的一部分。

        public static void NewEscapeSequence()
        {
            Console.WriteLine("[31m红色文本[0m");

            // C# 13 之前
            Console.WriteLine("\u001b[31m红色文本\u001b[0m"); //输出红色文字

            // C# 13 中
            Console.WriteLine("\e[31m红色文本\e[0m");//功能相同,语法更简洁
        }

 

方法组自然类型改进

此功能对涉及方法组的重载解析进行了少量优化。方法组是一个方法,并且所有重载都具有相同的名称。 编译器以前的行为是为方法组构造完整的候选方法集。如果需要自然类型,则自然类型是根据整套候选方法确定的。

  • 详细介绍:https://learn.microsoft.com/zh-cn/dotnet/csharp/language-reference/proposals/csharp-13.0/method-group-natural-type-improvements
// C# 13 中可以直接使用方法组并推断自然类型:
var a = Example.Method; // 成功推断为Action<int>(选择第一个匹配的委托类型)

// 通过上下文进一步优化推断:
List<Action<int>> actions = new() { Example.Method }; // 根据集合类型推断为Action<int>

隐式索引访问

在C# 13中允许在对象初始化表达式中使用^ 运算符(从末尾索引)直接为集合元素赋值。

        public class Numbers
        {
            public int[] Datas { get; set; } = new int[8];
        }

        public static void ImplicitIndexAccess()
        {
            var countdown = new Numbers()
            {
                Datas =
                {
                   [1] = 0,
                   [2] = 1,
                    
                   // 从 C# 13 开始可以执行下面方式赋值
                   [^3] = 2,
                   [^4] = 3,
                   [^5] = 4
                }
            };
        }

异步和迭代器方法中的 ref 与 unsafe 支持

在 C# 13 之前,迭代器方法(使用 yield return 的方法)和 async 方法不能声明局部 ref 变量,也不能使用 unsafe 上下文。

在 C# 13 中,async 方法可以声明 ref 局部变量或 ref struct 类型的局部变量。 但不可跨 await 或 yield return 使用。

同样,C# 13 允许在迭代器方法中使用 unsafe 上下文。但是,所有 yield return 和 yield break 语句都必须在安全的上下文中。

应用场景:提升内存敏感操作(如高性能 Span 处理)的灵活性。

allows ref struct

在 C# 13 之前,ref struct 类型不能声明为泛型或方法的类型参数。现在,泛型类型声明可以添加反约束 allows ref struct。 此反约束声明为该类型参数提供的类型参数可以是 ref struct 类型。编译器会对该类型参数的所有实例执行 ref 安全规则。

  • 应用场景:适用于游戏引擎、实时数据处理等需要低延迟内存操作的领域。通过泛型约束,可编写同时支持 ref struct 和非 ref struct 的通用代码。

例如,可以像下面的代码一样声明一个泛型类型:

public class C<T> where T : allows ref struct
{
    // 使用 T 作为 ref struct:
    public void M(scoped T p)
    {
        //参数 p 必须遵循 ref 安全规则 
    }
}

allows ref struct 反约束声明相应的类型参数可以是 ref struct 类型。 该类型参数的实例必须遵循以下规则:

  • 它不能被装箱。
  • 它参与引用安全规则。
  • 不能在不允许 ref struct 类型的地方使用实例,例如 static 字段。
  • 实例可以使用 scoped 修饰符进行标记。

partial类型中现在允许使用部分属性和索引器

现如今可以在 C# 13 中声明 partial属性partial索引器partial属性和索引器通常遵循与partial方法相同的规则:创建一个定义声明,一个实现声明。这两种声明的签名必须匹配。一个限制是,不能使用自动属性声明来实现部分属性。未声明正文的属性被视为声明声明。

注意:不允许对构造函数、终结器、重载运算符或事件声明使用 partial 关键字。在 C# 13 之前,不允许对属性或索引器使用partial。

public partial class MyClass
{
    public partial string Name { get; set; }
}

public partial class MyClass
{
    private string _name;
    public partial string Name
    {
        get => _name;
        set => _name = value;
    }
}

重载解析优先级

在 C# 13 中,编译器识别 OverloadResolutionPriorityAttribute,以便优先选择一个重载而不是另一个。库作者可以使用该属性确保新的、更好的重载比现有的重载更受青睐。

应用场景:适用于解决特定场景下的重载冲突和性能优化需求。通过合理设置优先级,开发者可以在保持代码兼容性的同时,优化编译器的选择逻辑。

    public class Printer
    {
        [OverloadResolutionPriority(1)] //优先调用
        public static void PrintWay(params int[] numberList) { }

        public static void PrintWay(params ReadOnlySpan<int> numberList) { }
    }

参考文章

  • https://learn.microsoft.com/zh-cn/dotnet/csharp/whats-new/csharp-13

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/907449.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Open R1 项目进展第一期

DeepSeek R1 发布已经两周了,而我们启动 open-r1 项目——试图补齐它缺失的训练流程和合成数据——也才过了一周。这篇文章简单聊聊:Open-R1 在模仿 DeepSeek-R1 流程和数据方面的进展 我们对 DeepSeek-R1 的认识和相关讨论 DeepSeek-R1 发布后社区搞出来的有趣项目这既是项目…

GPU内核实现(下)

3. ELLPACK 内核 ELLPACK SpMV实现沿行并行计算。由于数据已被重新排序为以列为主存储,因此沿ELLPACK数据连续行的内存访问被合并。在下面显示的实现中,假设输入cols和vals数组已经转换为ELLPACK格式。这种格式的一个关键部分是元数据参数,即每行非零的最大数量,它也作为参…

GPU内核实现(上)

GPU内核实现 以下是基于CSR和ELLPACK格式的一些标准SpMV实现。 1. 标量CSR内核 GPU加速SpMV的最简单实现之一是标量内核方法。标量内核分配一个线程来处理SpMV中的每个稀疏点积。稀疏点积由每个线程以顺序方式处理,从而消除了对需要共享内存和/或扭曲级别降低的更高级技术的需…

稀疏矩阵向量乘法介绍

稀疏矩阵向量乘法介绍 稀疏矩阵向量乘法(SpMV)是每个隐式稀疏线性代数求解器。从简单的 Krylov 算法到 multigrid 的算法性能方法在很大程度上取决于 SpMV 实现的速度。因为 SpMV 具有非常低的算术强度,定义为浮点操作数,则实现速度受内存带宽。最大化内存带宽的实现将实现…

推荐关注《AI芯片开发核心技术详解》(1)、《智能汽车传感器:原理设计应用》(2)、《TVM编译器原理与实践》(3)、《LLVM编译器原理与实践》(4)

4本书推荐《AI芯片开发核心技术详解》、《智能汽车传感器:原理设计应用》、《TVM编译器原理与实践》、《LLVM编译器原理与实践》由清华大学出版社资深编辑赵佳霓老师策划编辑的新书《AI芯片开发核心技术详解》已经出版,京东、淘宝天猫、当当等网上,相应陆陆续续可以购买。该…

Ollama+OneAPI+Open WebUI 搭建本地大模型

✅Ollama 安装 ✅极简安装 curl -fsSL https://ollama.com/install.sh | sh✅Docker 安装 ❗前提是已安装NVIDIA Container Toolkit # 拉取镜像 docker pull ollama/ollama# 启动容器 docker run -d --gpus=all -v ollama:/root/.ollama -p 11434:11434 --name ollama ollama/o…

洛谷 P1216 [IOI 1994] 数字三角形 Number Triangles (记忆化搜索)

记忆化搜索思路:经典的DP题,看题解大佬个个是状态转移方程...我就写个记忆化搜索吧,这个数据量,只dfs暴搜是过不去的,写完记忆化之后发现有个测试点T了,下载了一波测试点数据,发现全是0,那么初始化dp数组为-1就好了。AcCode: #include<bits/stdc++.h> using nam…

【软件】在Windows和Ubuntu上使用TFTP和NFS

在Windows和Ubuntu上使用TFTP和NFS 零、介绍 最近在玩Linux开发板,在开发的过程中发现需要用到tftp和nfs来帮助传输文件,故此记录如何使用这两种软件。 TFTP(Trivial File Transfer Protocol) :是一种简化的文件传输协议,设计用于在客户端和服务器之间快速传输文件。轻量…

FastAPI Pydantic动态调整Schema

title: FastAPI Pydantic动态调整Schema date: 2025/3/29 updated: 2025/3/29 author: cmdragon excerpt: Pydantic动态Schema支持运行时字段调整和环境变量控制,实现毫秒级配置生效。通过字段级动态注入和条件必填验证,灵活适应业务需求。多租户系统采用条件字段过滤实现数…

【杭电多校比赛记录】2025“钉耙编程”中国大学生算法设计春季联赛(4)

比赛链接 本文发布于博客园,会跟随补题进度实时更新,若您在其他平台阅读到此文,请前往博客园获取更好的阅读体验。 跳转链接:https://www.cnblogs.com/TianTianChaoFangDe/p/18799072 开题 + 补题情况 和前三场比起来前期的签到题发挥稳定了许多,没有被卡很久,不过 1001 …

详细介绍Mybatis的缓存机制

一、缓存机制 1、缓存概述 缓存:缓存就是一块内存空间,保存临时数据 作用:将数据源(数据库或者文件)中的数据读取出来存放到缓存中,再次获取时直接从缓存中获取,可以减少和数据库交互的次数,提升程序的性能 缓存适用: 适用于缓存的:经常查询但不经常修改的,数据的正…

JS梳理之es6异步async await 协程 迭代器

es6异步 promise 链式调用 是对回调炼狱的一种优化 这次梳理一下async await async function fetchData() {const response = await fetch(https://api.example.com/data);const data = await response.json();return data; }// async 声明这是一个异步函数 // await 会暂停函数…