net核心中使用ref和Span提高程序性能的方法

  介绍

这篇文章给大家分享的是有关。网络核心中使用ref和Span提高程序性能的方法的内容。小编觉得挺实用的,因此分享给大家做个参考。一起跟随小编过来看看吧。

其实说到裁判,很多同学对它已经有所了解,裁判是c# 7.0的一个语言特性,它为开发人员提供了返回本地变量引用和值引用的机制。
跨度也是建立在裁判语法基础上的一个复杂的数据类型,在文章的后半部分,我会有一个例子说明如何使用它。

不论是裁判还是出关键,都是一种比较难以理解和操作的语言特性,如C语言中操作指针一样,这样的高级语法总是什么带来一些副作用,但是我不认为这有什么,而且不是每一个c#开发者都要对这些内部运行的机制有着深刻的理解,我觉得不论什么复杂的东西只是为人们提供了一个自由的选择,风险和灵活性永远是不能兼容的。

来看几个例子来说明引用与指针的相同性,当然下面的使用方式早在c# 7.0之前就可以使用了:

公共静态孔隙IncrementByRef (ref int x)
  {
  x + +;
  }
  公共安全静态孔隙IncrementByPointer (int * x)
  {
  (* x) + +;
  }

上面两个函数分别是使用ref和非安全指针来完成参数+ 1。

 int i=30;
  IncrementByRef (ref);//i=31
  不安全的{
  IncrementByPointer(和我);
  }//i=32 

下面是c# 7.0提供的特性:

<强> 1。当地人(ref引用本地变量)

 int i=42;
  ref var x=ref我;
  x=x + 1;//i=

这43个例子中为本我变地量的引用x,当改变x的值时我变量的值也改变了。

<强> 2。ref回报(返回值引用)

ref回报是c# 7中一个强大的特性,下面代码是最能体现其特性的,该函数提供了,返回int数组中某一项的引用:

公共静态ref int GetArrayRef (int[]项目,int指数)=比;ref物品(指数);

通过下标取得数组中的项目的引用,改变引用值时,数组也会随之改变。

System.Span是。网络核心的核心的一部分,在System.Memory。dll程序集下。目前该特性是独立的,将来可能会集成到CoreFx中;

如何使用呢?在。网络核心2.0 SDK创建的项目下引用如下NuGet包:

 & lt; ItemGroup>
  & lt; PackageReference包括=癝ystem.Memory"Version=?.4.0-preview1-25305-02"/比;
  & lt; PackageReference包括=癝ystem.Runtime.CompilerServices.Unsafe"Version=?.4.0-preview1-25305-02"/比;
  & lt;/ItemGroup> 

在上面我们看到了使用ref关键字可以提供的类似指针(T *)的操作单一值对象方式。基本上在。net体系下操作指针都不认为是一件好的事件,当然。net为我们提供了安全操作单值引用的ref。但是单值只是用户使用”指针”的一小部分需求,对于指针来说,更常见的情况是操作一系列连续的内存空间中的“元素”时。

跨度表示为一个已知长度和类型的连续内存块。许多方面讲它非常类似T[]或ArraySegment,它提供安全的访问内存区域指针的能力,其实我理解它更将是。net中操作(void *)指针的抽象,熟悉C/c++开发者应该更明白这意味着什么。

?抽象了所有连续内存空间的类型系统,包括:数组,非托管指针,堆栈指针,固定或固定过的托管数据,以及值内部区域的引用
?支持CLR标准对象类型和值类型
?支持泛型
?支持GC,而不像指针需要自己来管理释放

下面来看下跨度的定义,它与裁判有着语法和语义上的联系:

字符串内容=澳谌莩ざ?123“;;
  秒表watch2=new秒表();
  watch2.Start ();
  for (int j=0;j & lt;100000;j + +)
  {
  int.Parse (content.Substring (15));
  }
  watch2.Stop ();
  Console.WriteLine (“\ tTime运行:\ t"+ watch2.ElapsedMilliseconds.ToString (“N0") +“ms"); 

为什么使用这个例子呢,这是一个典型的子串的使用场景,每次操作字符串都会生成新字符串的对象,当然不光是子字符串,在进行int。解析时重复操作字符串对象,如果大量操作就会给GC造成压力。

net核心中使用ref和Span提高程序性能的方法