詳解c# 切片語法糖
一:背景
1. 講故事
昨天在 github 上準(zhǔn)備找找 C# 9 又有哪些新語法糖可以試用,不覺在一個(gè)文檔上看到一個(gè)很奇怪的寫法: foreach (var item in myArray[0..5])
哈哈,熟悉又陌生,玩過python的朋友對這個(gè) [0..5] 太熟悉不過了,居然在 C# 中也遇到了,開心哈,看了下是 C# 8 的新語法,諷刺諷刺,8 都沒玩熟就搞 9 了,我的探索欲比較強(qiáng),總想看看這玩意底層是由什么支撐的。
二:.. 語法糖的用法
從前面介紹的 myArray[0..5]
語義上也能看出,這是一個(gè)切分array的操作,那到底有幾種切分方式呢? 下面一個(gè)一個(gè)來介紹,為了方便演示,我先定義一個(gè)數(shù)組,代碼如下:
var myarr = new string[] { "10", "20", "30", "40", "50", "60", "70", "80", "90", "100" };
1. 提取 arr 前3個(gè)元素
如果用 linq 的話,可以用 Take(3),用切片操作的話就是 [0..3], 代碼如下:
static void Main(string[] args) { var myarr = new string[] { "10", "20", "30", "40", "50", "60", "70", "80", "90", "100" }; //1. 獲取數(shù)組 前3個(gè)元素 var query1 = myarr[0..3]; var query2 = myarr.Take(3).ToList(); Console.WriteLine($"query1={string.Join(",", query1)}"); Console.WriteLine($"query2={string.Join(",", query2)}"); }
2. 提取 arr 最后三個(gè)元素
這個(gè)怎么提取呢?在 python 中直接用 -3 表示就可以了,在C# 中需要用 ^ 來表示從末尾開始,代碼如下:
static void Main(string[] args) { var myarr = new string[] { "10", "20", "30", "40", "50", "60", "70", "80", "90", "100" }; //1. 獲取數(shù)組 最后3個(gè)元素 var query1 = myarr[^3..]; var query2 = myarr.Skip(myarr.Length - 3).ToList(); Console.WriteLine($"query1={string.Join(",", query1)}"); Console.WriteLine($"query2={string.Join(",", query2)}"); }
3. 提取 array 中index = 4,5,6 的三個(gè)位置元素
用 linq 的話,就需要使用 Skip + Take
雙組合,如果用切片操作的話就太簡單了。。。
static void Main(string[] args) { var myarr = new string[] { "10", "20", "30", "40", "50", "60", "70", "80", "90", "100" }; //1. 獲取數(shù)組 中 index=4,5,6 三個(gè)位置的元素 var query1 = myarr[4..7]; var query2 = myarr.Skip(4).Take(3).ToList(); Console.WriteLine($"query1={string.Join(",", query1)}"); Console.WriteLine($"query2={string.Join(",", query2)}"); }
從上面的切割區(qū)間 [4..7]
的輸出結(jié)果來看,這是一個(gè) 左閉右開 的區(qū)間,所以要特別注意一下。
4. 獲取 array 中倒數(shù)第三和第二個(gè)元素
從要求上來看就是獲取元素 80 和 90,如果你理解了前面的兩個(gè)用法,我相信這個(gè)你會很快的寫出來,代碼如下:
static void Main(string[] args) { var myarr = new string[] { "10", "20", "30", "40", "50", "60", "70", "80", "90", "100" }; //1. 獲取 array 中倒數(shù)第三和第二個(gè)元素 var query1 = myarr[^3..^1]; var query2 = myarr.Skip(myarr.Length - 3).Take(2).ToList(); Console.WriteLine($"query1={string.Join(",", query1)}"); Console.WriteLine($"query2={string.Join(",", query2)}"); }
三. 探究原理
通過前面 4 個(gè)例子,我想大家都知道怎么玩了,接下來就是看看到底內(nèi)部是用什么做支撐的,這里使用 DnSpy 去挖挖看。
1. 從 myarr[0..3] 看起
用 dnspy 反編譯代碼如下:
//編譯前 var query1 = myarr[0..3]; //編譯后: string[] query = RuntimeHelpers.GetSubArray<string>(myarr, new Range(0, 3));
從編譯后的代碼可以看出,原來獲取切片的 array 是調(diào)用 RuntimeHelpers.GetSubArray
得到了,然后我簡化一下這個(gè)方法,代碼如下:
public static T[] GetSubArray<[Nullable(2)] T>(T[] array, Range range) { ValueTuple<int, int> offsetAndLength = range.GetOffsetAndLength(array.Length); int item = offsetAndLength.Item1; int item2 = offsetAndLength.Item2; T[] array3 = new T[item2]; Buffer.Memmove<T>(Unsafe.As<byte, T>(array3.GetRawSzArrayData()), Unsafe.Add<T>(Unsafe.As<byte, T>(array.GetRawSzArrayData()), item), (ulong)item2); return array3; }
從上面代碼可以看到,最后的 子array 是由 Buffer.Memmove
完成的,但是給 子array 的切割位置是由 GetOffsetAndLength
方法實(shí)現(xiàn),繼續(xù)追一下代碼:
public readonly struct Range : IEquatable<Range> { public Index Start { get; } public Index End { get; } public Range(Index start, Index end) { this.Start = start; this.End = end; } public ValueTuple<int, int> GetOffsetAndLength(int length) { Index start = this.Start; int num; if (start.IsFromEnd) { num = length - start.Value; } else { num = start.Value; } Index end = this.End; int num2; if (end.IsFromEnd) { num2 = length - end.Value; } else { num2 = end.Value; } return new ValueTuple<int, int>(num, num2 - num); } }
看完上面的代碼,你可能有兩點(diǎn)疑惑:
1) start.IsFromEnd 和 end.IsFromEnd 是什么意思。
其實(shí)看完上面代碼邏輯,你就明白了,IsFromEnd 表示起始點(diǎn)是從左開始還是從右邊開始,就這么簡單。
2) 我并沒有看到 start.IsFromEnd 和 end.IsFromEnd 是怎么賦上值的。
在 Index 類的構(gòu)造函數(shù)中,取決于上一層怎么去 new Index 的時(shí)候塞入的 true 或者 false,如下代碼:
這個(gè)例子的流程大概是: new Range(1,3) -> operator Index(int value) -> FromStart(value) -> new Index(value)
,可以看到最后在 new 的時(shí)候并沒有對可選參數(shù)賦值。
2. 探究 myarr[^3..]
剛才的例子是沒有對可選參數(shù)賦值,那看看本例是不是 new Index 的時(shí)候賦值了?
//編譯前: var query1 = myarr[^3..]; //編譯后: string[] query = RuntimeHelpers.GetSubArray<string>(myarr, Range.StartAt(new Index(3, true)));
看到?jīng)]有,這一次 new Index 的時(shí)候,給了 IsFromEnd = true , 表示從末尾開始計(jì)算,大家再結(jié)合剛才的 GetOffsetAndLength 方法,我想這邏輯你應(yīng)該理順了吧。
四:總結(jié)
總的來說這個(gè)切片操作太實(shí)用了,作用于 arr 可以大幅度減少對 skip & take 的使用,作用于 string 也可以大幅減少 SubString 的使用,如:"12345"[1..3] -> "12345".Substring(1, 2)
,嘿嘿,厲害了吧! 還是C# 大法🐂👃
更多高質(zhì)量干貨:參見我的 GitHub: dotnetfly
以上就是詳解c# 切片語法糖的詳細(xì)內(nèi)容,更多關(guān)于c# 切片的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C#實(shí)現(xiàn)判斷操作系統(tǒng)是否為Win8以上版本
這篇文章主要介紹了C#實(shí)現(xiàn)判斷操作系統(tǒng)是否為Win8以上版本,本文講解了利用C#獲取OS的版本號、利用反射獲取當(dāng)前正在運(yùn)行的程序的版本信息、 利用C#判斷當(dāng)前操作系統(tǒng)是否為Win8系統(tǒng)等內(nèi)容,需要的朋友可以參考下2015-06-06C#制作鷹眼的詳細(xì)全過程(帶注釋)實(shí)例代碼
C#制作鷹眼的詳細(xì)全過程(帶注釋)實(shí)例代碼,需要的朋友可以參考一下2013-03-03詳解c# 強(qiáng)制轉(zhuǎn)換和類型轉(zhuǎn)換
這篇文章主要介紹了c# 強(qiáng)制轉(zhuǎn)換和類型轉(zhuǎn)換的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)c#,感興趣的朋友可以了解下2020-10-10C#常用多線程(線程同步,事件觸發(fā),信號量,互斥鎖,共享內(nèi)存,消息隊(duì)列)
這篇文章主要介紹了C#常用多線程(線程同步,事件觸發(fā),信號量,互斥鎖,共享內(nèi)存,消息隊(duì)列),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-09-09linux操作系統(tǒng)安裝MONO執(zhí)行C#程序的詳解步驟
這篇文章主要介紹了linux操作系統(tǒng)安裝MONO執(zhí)行C#程序詳解步驟,有需要的可以參考一下2013-12-12Unity?制作一個(gè)分?jǐn)?shù)統(tǒng)計(jì)系統(tǒng)
項(xiàng)目中經(jīng)常遇到分?jǐn)?shù)統(tǒng)計(jì)的需求,例如操作正確則計(jì)分,相反則不計(jì)分失去該項(xiàng)分?jǐn)?shù),為了應(yīng)對需求需要一個(gè)分?jǐn)?shù)統(tǒng)計(jì)系統(tǒng)。本文主要介紹了通過Unity實(shí)現(xiàn)這樣的一個(gè)計(jì)分系統(tǒng),快來跟隨小編一起學(xué)習(xí)吧2021-12-12