昨天小桂问了一个问题,把一个数组的全部元素加1,有什么好办法,于是有了下面的分析:
var arr = new int[] { 1, 2, 3, 4, 5, 6, 7, 8, 9 };
//方法一
foreach (var i in arr)
{i++;
}
//方法二
for (var i = 0; i < arr.Length; i++)
{arr[i] += 1;
}
//方法三
foreach (ref var i in arr.AsSpan())
{i++;
}
方法一在编译时会报错的,提示如下:无法为"i"赋值,因为它是"foreach"迭代变量,foreach循环时,是通过yield return返回的只读数据,所以不能给它赋值。
方法二没有问题,遍历元素,轮询加1。
方法三比较有意思了,是通过Span<T>来用ref(可以理解成指针,或引用)的方式来,来指向数组的元素,实现加1。
关于Span<T>,借用官方的一句话:“由于 Span<T> 是任意内存块(可以是托管内存,本机内存,堆栈内存)的抽象,因此 Span<T> 具有参数的类型和方法的方法将 Span<T> 在任何对象上操作, Span<T> 而不考虑它所封装的内存类型。”,可以清楚的理解Span<T>的专门作用。
关于方法二和方法三的性能怎么样,下来是测试代码和结果,供大家参考:
using BenchmarkDotNet.Attributes;
using BenchmarkDotNet.Running;BenchmarkRunner.Run<TestSpan>();public class TestSpan
{[Benchmark]public void Demo1(){var arr = new int[] { 1, 2, 3, 4, 5, 6, 7, 8, 9 };foreach (ref var i in arr.AsSpan()){i++;}}[Benchmark]public void Demo2(){var arr = new int[] { 1, 2, 3, 4, 5, 6, 7, 8, 9 };for (var i = 0; i < arr.Length; i++){arr[i] += 1;}}
}
结果: