C# 中居然也有切片语法糖,太厉害了

从前面介绍的 myArray[0..5] 语义上也能看出,这是一个切分array的操作,那到底有几种切分方式呢? 下面一个一个来介绍,为了方便演示,我先定义一个数组,代码如下:

var myarr =newstring[] {"10","20","30","40","50","60","70","80","90","100"};

1. 提取 arr 前3个元素

如果用 linq 的话,可以用 Take(3),用切片操作的话就是 [0..3], 代码如下:

staticvoidMain(string[] args){            var myarr =newstring[] {"10","20","30","40","50","60","70","80","90","100"};//1. 获取数组 前3个元素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 最后三个元素

这个怎么提取呢?在 python 中直接用 -3 表示就可以了,在C# 中需要用 ^ 来表示从末尾开始,代码如下:

staticvoidMain(string[] args){            var myarr =newstring[] {"10","20","30","40","50","60","70","80","90","100"};//1. 获取数组 最后3个元素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 的三个位置元素

用 linq 的话,就需要使用 Skip + Take 双组合,如果用切片操作的话就太简单了。。。

staticvoidMain(string[] args){            var myarr =newstring[] {"10","20","30","40","50","60","70","80","90","100"};//1. 获取数组 中 index=4,5,6 三个位置的元素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)}");        }

从上面的切割区间 [4..7] 的输出结果来看,这是一个 左闭右开 的区间,所以要特别注意一下。

4. 获取 array 中倒数第三和第二个元素

从要求上来看就是获取元素 80 和 90,如果你理解了前面的两个用法,我相信这个你会很快的写出来,代码如下:

staticvoidMain(string[] args){            var myarr =newstring[] {"10","20","30","40","50","60","70","80","90","100"};//1. 获取 array 中倒数第三和第二个元素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 个例子,我想大家都知道怎么玩了,接下来就是看看到底内部是用什么做支撑的,这里使用 DnSpy 去挖挖看。

1. 从 myarr[0..3] 看起

用 dnspy 反编译代码如下:

//编译前var query1 = myarr[0..3];//编译后:string[] query = RuntimeHelpers.GetSubArray(myarr,newRange(0,3));

从编译后的代码可以看出,原来获取切片的 array 是调用 RuntimeHelpers.GetSubArray 得到了,然后我简化一下这个方法,代码如下:

publicstaticT[] GetSubArray<[Nullable(2)] T>(T[]array, Range range)        {            ValueTuple offsetAndLength = range.GetOffsetAndLength(array.Length);intitem = offsetAndLength.Item1;intitem2 = offsetAndLength.Item2;            T[] array3 =newT[item2];            Buffer.Memmove(Unsafe.As(array3.GetRawSzArrayData()), Unsafe.Add(Unsafe.As(array.GetRawSzArrayData()), item), (ulong)item2);returnarray3;        }

从上面代码可以看到,最后的 子array 是由 Buffer.Memmove 完成的,但是给 子array 的切割位置是由 GetOffsetAndLength 方法实现,继续追一下代码:

publicreadonlystructRange:IEquatable    {publicIndex Start { get; }publicIndex End { get; }publicRange(Index start, Index end){this.Start = start;this.End = end;}publicValueTuple GetOffsetAndLength(intlength)        {            Index start =this.Start;intnum;if(start.IsFromEnd)            {                num = length - start.Value;            }else{                num = start.Value;            }            Index end =this.End;intnum2;if(end.IsFromEnd)            {                num2 = length - end.Value;            }else{                num2 = end.Value;            }returnnewValueTuple(num, num2 - num);        }    }

深圳网站建设www.sz886.com

你可能感兴趣的:(C# 中居然也有切片语法糖,太厉害了)