C# 12 中新增的幾大功能你都知道嗎?
轉(zhuǎn)眼之間C#都已經(jīng)更新到了12了,那么C# 12 中新增的八大功能大家都了解過(guò)嗎?今天我們來(lái)簡(jiǎn)單講解一下C# 12 中的八大新增功能。
一、主構(gòu)造函數(shù)
在 Visual Studio 2022 版本 17.6 預(yù)覽版 2 中引入。
從 C# 12 開(kāi)始,可以在類和結(jié)構(gòu)中聲明主構(gòu)造函數(shù)。主構(gòu)造函數(shù)參數(shù)都在類的整個(gè)主體的范圍內(nèi)。 為了確保顯式分配所有主構(gòu)造函數(shù)參數(shù),所有顯式聲明的構(gòu)造函數(shù)都必須使用 this() 語(yǔ)法調(diào)用主構(gòu)造函數(shù)。 將主構(gòu)造函數(shù)添加到 class 可防止編譯器聲明隱式無(wú)參數(shù)構(gòu)造函數(shù)。 在 struct 中,隱式無(wú)參數(shù)構(gòu)造函數(shù)初始化所有字段,包括 0 位模式的主構(gòu)造函數(shù)參數(shù)。
1.主構(gòu)造函數(shù)參數(shù)的最常見(jiàn)用途包括:
- 作為 base() 構(gòu)造函數(shù)調(diào)用的參數(shù)。
- 初始化成員字段或?qū)傩浴?/li>
- 引用實(shí)例成員中的構(gòu)造函數(shù)參數(shù)。
2.代碼示例
將任何參數(shù)放在類型名稱后面的括號(hào)中:
public class NameParameter(string name)
{
public string Name => name;
}
以下代碼初始化從主構(gòu)造函數(shù)參數(shù)計(jì)算的兩個(gè)只讀屬性:
public readonly struct Distance(double dx, double dy)
{
public readonly double Magnitude { get; } = Math.Sqrt(dx * dx + dy * dy);
public readonly double Direction { get; } = Math.Atan2(dy, dx);
}
二、集合表達(dá)式
在 Visual Studio 2022 版本 17.7 預(yù)覽版 5 中引入。
集合表達(dá)式引入了一種新的簡(jiǎn)潔語(yǔ)法,用于創(chuàng)建常用集合值。可以使用展開(kāi)運(yùn)算符(..)將其他集合內(nèi)聯(lián)到這些值中。
1.下面的示例展示了集合表達(dá)式的用法:
// Create an array
int[] array = [55, 99, 100, 33];
// Create a list
List<string> list = ["one", "two", "three", "five", "追逐時(shí)光者"];
// Create a span
Span<char> c = ['a', 'b', 'c', 'd', 'e', 'f', 'h', 'i', 'k'];
// Create a jagged 2D array
int[][] two2D = [[1, 2, 3], [4, 5, 6], [7, 8, 9], [88, 8, 9]];
2.展開(kāi)運(yùn)算符(..)示例代碼:
展開(kāi)運(yùn)算符(集合表達(dá)式中的 ..)可將其參數(shù)替換為該集合中的元素。 參數(shù)必須是集合類型。 以下示例演示了展開(kāi)運(yùn)算符的工作原理:
int[] item0 = [88, 2, 3];
int[] item1 = [22, 5, 6];
int[] item2 = [7, 99, 9];
int[] single = [.. item0, .. item1, .. item2];
foreach (var element in single)
{
Console.Write($"{element}, ");
}
沒(méi)有.. 會(huì)有異常:
正常輸出:
三、內(nèi)聯(lián)數(shù)組
在 Visual Studio 2022 版本 17.7 預(yù)覽版 3 中引入。
運(yùn)行時(shí)團(tuán)隊(duì)和其他庫(kù)作者使用內(nèi)聯(lián)數(shù)組來(lái)提高應(yīng)用程序的性能。通過(guò)內(nèi)聯(lián)數(shù)組,開(kāi)發(fā)人員可以在結(jié)構(gòu)類型中創(chuàng)建固定大小的數(shù)組。具有內(nèi)聯(lián)緩沖區(qū)的結(jié)構(gòu)體應(yīng)具有與不安全固定大小緩沖區(qū)類似的性能特性。
內(nèi)聯(lián)數(shù)組的聲明與下面的結(jié)構(gòu)類似:
[System.Runtime.CompilerServices.InlineArray(20)]
public struct Buffer
{
private int _element0;
}
你可以像使用其他數(shù)組一樣使用它們:
public static void Test()
{
var buffer = new Buffer();
for (int i = 0; i < 20; i++)
{
buffer[i] = i;
}
foreach (var i in buffer)
{
Console.WriteLine(i);
}
}
四、Lambda 表達(dá)式中的可選參數(shù)
在 Visual Studio 2022 版本 17.5 預(yù)覽版 2 中引入。
可以為 Lambda 表達(dá)式的參數(shù)定義默認(rèn)值。 語(yǔ)法和規(guī)則與將參數(shù)的默認(rèn)值添加到任何方法或本地函數(shù)相同。
如果 lambda 表達(dá)式只有一個(gè)輸入?yún)?shù),則括號(hào)是可選的:
Func<double, double> cube = x => x * x * x;
兩個(gè)或更多輸入?yún)?shù)使用逗號(hào)加以分隔:
Func<int, int, bool> testForEquality = (x, y) => x == y;
可以顯式指定類型,如下面的示例所示:
Func<int, string, bool> isTooLong = (int x, string s) => s.Length > x;
注意:輸入?yún)?shù)類型必須全部為顯式或全部為隱式;否則,便會(huì)生成 CS0748 編譯器錯(cuò)誤!!
五、ref readonly參數(shù)
在 Visual Studio 2022 版本 17.8 預(yù)覽版 2 中引入。
ref readonly修飾符表示方法希望參數(shù)是一個(gè)變量,而不是一個(gè)非變量的表達(dá)式。不是變量的表達(dá)式包括常量、方法返回值和屬性。如果參數(shù)不是變量,編譯器會(huì)發(fā)出警告。
六、任何類型的別名
在 Visual Studio 2022 版本 17.6 預(yù)覽版 3 中引入。
可以使用 using alias 指令來(lái)別名任何類型,而不僅僅是命名類型。也就是說(shuō),你可以為元組類型、數(shù)組類型、指針類型或其他不安全類型創(chuàng)建語(yǔ)義別名。
使用 using 關(guān)鍵字為元組類型創(chuàng)建別名,并進(jìn)行調(diào)用:
using PointTest = (int x, int y);
namespace Csharp12
{
internal class Class1
{
public static void Test()
{
PointTest point = (10, 20);
Console.WriteLine($"Point coordinates: X={point.Item1}, Y={point.Item2}");
}
}
}
七、Experimental屬性
在 Visual Studio 2022 版本 17.7 預(yù)覽版 3 中引入。
可以使用 System.Diagnostics.CodeAnalysis.ExperimentalAttribute 來(lái)標(biāo)記類型、方法或程序集,以指示實(shí)驗(yàn)性特征。 如果訪問(wèn)使用 ExperimentalAttribute 注釋的方法或類型,編譯器將發(fā)出警告。 用 Experimental 特性標(biāo)記的程序集中包含的所有類型都是實(shí)驗(yàn)性的。
示例代碼:
namespace Csharp12
{
[AttributeUsage(System.AttributeTargets.Assembly | System.AttributeTargets.Class | System.AttributeTargets.Constructor | System.AttributeTargets.Delegate | System.AttributeTargets.Enum | System.AttributeTargets.Event | System.AttributeTargets.Field | System.AttributeTargets.Interface | System.AttributeTargets.Method | System.AttributeTargets.Module | System.AttributeTargets.Property | System.AttributeTargets.Struct)]
public class ExperimentalAttribute : Attribute
{
public ExperimentalAttribute()
{
}
}
[Experimental]
public class ExperimentalClass
{
public void DoSomething()
{
Console.WriteLine("Doing something experimental...");
}
}
internal class Class1
{
public static void Test()
{
ExperimentalClass exp = new ExperimentalClass();
exp.DoSomething();
}
}
}
八、攔截器
預(yù)覽功能在 Visual Studio 2022 版本 17.7 預(yù)覽版 3 中引入。
攔截器是一種方法,該方法可以在編譯時(shí)以聲明方式將對(duì)可攔截方法的調(diào)用替換為對(duì)其自身的調(diào)用。 通過(guò)讓攔截器聲明所攔截調(diào)用的源位置,可以進(jìn)行這種替換。 攔截器可以向編譯中(例如在源生成器中)添加新代碼,從而提供更改現(xiàn)有代碼語(yǔ)義的有限能力。
注意:攔截器是一項(xiàng)試驗(yàn)性功能,在 C# 12 的預(yù)覽模式下提供。 在將來(lái)的版本中,該功能可能會(huì)發(fā)生中斷性變更或被刪除。 因此,不建議將其用于生產(chǎn)或已發(fā)布的應(yīng)用程序。
要使用攔截器,用戶項(xiàng)目必須指定 <InterceptorsPreviewNamespaces> 屬性。這是允許包含攔截器的命名空間列表。
<InterceptorsPreviewNamespaces>$(InterceptorsPreviewNamespaces);Microsoft.AspNetCore.Http.Generated;MyLibrary.Generated</InterceptorsPreviewNamespaces>