全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

.Net Core中使用ref和Span<T>提高程序性能的实现代码

一、前言

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

二、ref关键字

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

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

public static void IncrementByRef(ref int x)
{
 x++;
}
public unsafe static void IncrementByPointer(int* x)
{
 (*x)++;
}

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

int i = 30;
IncrementByRef(ref i);
// i = 31
unsafe{
 IncrementByPointer(&i);
}
// i = 32

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

1.ref locals (引用本地变量)

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

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

2.ref returns (返回值引用)

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

public static ref int GetArrayRef(int[] items, int index) => ref items[index];

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

三、Span

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

如何使用呢?在.Net Core 2.0 SDK创建的项目下引用如下NuGet包:

 <ItemGroup>
 <PackageReference Include="System.Memory" Version="4.4.0-preview1-25305-02" />
 <PackageReference Include="System.Runtime.CompilerServices.Unsafe" Version="4.4.0-preview1-25305-02" />
 </ItemGroup>

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

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

Span的特点如下:

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

下面来看下Span的定义,它与ref有着语法和语义上的联系:

public struct Span<T> {
 ref T _reference;
 int _length;
 public ref T this[int index] { get {...} }
 ...
}
public struct ReadOnlySpan<T> {
 ref T _reference;
 int _length;
 public T this[int index] { get {...} }
 ...
}

接下来我会用一个直观的例子来说明Span的使用场景;我们以字符截取和字符转换(转换为整型)为例:

如有一个字符串string content = "content-length:123",要转换将123转换为整型,通常的做法是先Substring将与数字字符无关的字符串进行截断,转换代码如下:

string content = "content-length:123";
Stopwatch watch1 = new Stopwatch();
watch1.Start();
for (int j = 0; j < 100000; j++)
{
 int.Parse(content.Substring(15));
}
watch1.Stop();
Console.WriteLine("\tTime Elapsed:\t" + watch1.ElapsedMilliseconds.ToString("N0") + "ms");

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

使用Span实现这个算法:

string content = "content-length:123";
ReadOnlySpan<char> span = content.ToCharArray(); 
span.Slice(15).ParseToInt();
watch.Start();
for (int j = 0; j < 100000; j++)
{
 int icb = span.Slice(15).ParseToInt();
}
watch.Stop();
Console.WriteLine("\tTime Elapsed:\t" + watch.ElapsedMilliseconds.ToString("N0") + "ms");

这里将string转换为int的算法利用ReadonlySpan实现,这也是Span的典型使用场景,官方给的场景也是如些,Span适用于多次复用操作连续内存的场景。

转换代码如下:

public static class ReadonlySpanxtension
{
 public static int ParseToInt(this ReadOnlySpan<char> rspan)
 {
  Int16 sign = 1;
  int num = 0;
  UInt16 index = 0;
  if (rspan[0].Equals('-')){
   sign = -1; index = 1;
  }
  for (int idx = index; idx < rspan.Length; idx++){
   char c = rspan[idx];
   num = (c - '0') + num * 10;
  }
  return num * sign;
 }
}

四、最后

上述两段代码100000次调用的时间如下:

String Substring Convert:
  Time Elapsed: 18ms
ReadOnlySpan Convert:
  Time Elapsed: 4ms

目前Span的相关支持还够,它只是最基础架构,之后CoreFx会对很多API使用Span进行重构和实现。可见.Net Core的性能日后会越来越强大。

以上所述是小编给大家介绍的.Net Core中使用ref和Span<T>提高程序性能的方法,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!


# .net  # core  # 性能  # 记一次EFCore类型转换错误及解决方案  # EFCore 通过实体Model生成创建SQL Server数据库表脚本  # 从EFCore上下文的使用到深入剖析DI的生命周期最后实现自动属性注入  # 如何在Asp.Net Core中集成Refit  # .net core实用技巧——将EF Core生成的SQL语句显示在控制台中  # .net EF Core专题:EF Core 读取数据时发生了什么?  # .net core EF Core调用存储过程的方式  # 详解.Net Core 权限验证与授权(AuthorizeFilter、ActionFilterAt  # .NET Core类库System.Reflection.DispatchProxy实现简易Aop的  # 在.NET Core类库中使用EF Core迁移数据库到SQL Server的方法  # CodeFirst从零开始搭建Asp.Net Core2.0网站  # 详解EFCore中的导航属性  # 转换为  # 小编  # 组中  # 都是  # 整型  # 几个  # 我不  # 我会  # 都不  # 也会  # 我觉得  # 如有  # 基础上  # 适用于  # 说到  # 这是一个  # 将是  # 给大家  # 会对  # 早在 


相关文章: 建站VPS选购需注意哪些关键参数?  深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?  建站之星3.0如何解决常见操作问题?  建站之星代理平台如何选择最佳方案?  免费公司网站制作软件,如何申请免费主页空间做自己的网站?  赚钱网站制作软件,建一个网站怎样才能赚钱?是如何盈利的?  北京企业网站设计制作公司,北京铁路集团官方网站?  建站VPS能否同时实现高效与安全翻墙?  香港服务器WordPress建站指南:SEO优化与高效部署策略  ,网页ppt怎么弄成自己的ppt?  如何用景安虚拟主机手机版绑定域名建站?  建站之星如何快速生成多端适配网站?  建站之星官网登录失败?如何快速解决?  如何通过商城免费建站系统源码自定义网站主题?  已有域名和空间如何搭建网站?  网站制作外包价格怎么算,招聘网站上写的“外包”是什么意思?  大连网站制作公司哪家好一点,大连买房网站哪个好?  北京网站制作网页,网站升级改版需要多久?  建站之星代理费用多少?最新价格详情介绍  网站制作知乎推荐,想做自己的网站用什么工具比较好?  深圳网站制作的公司有哪些,dido官方网站?  如何基于云服务器快速搭建网站及云盘系统?  建站之星收费标准详解:套餐费用及年费价格表一览  建站ABC备案流程中有哪些关键注意事项?  制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?  如何在Windows服务器上快速搭建网站?  建站之星多图banner生成与模板自定义指南  制作表格网站有哪些,线上表格怎么弄?  成都网站制作报价公司,成都工业用气开户费用?  如何在云主机快速搭建网站站点?  ,如何利用word制作宣传手册?  建站一年半SEO优化实战指南:核心词挖掘与长尾流量提升策略  SAX解析器是什么,它与DOM在处理大型XML文件时有何不同?  如何选购建站域名与空间?自助平台全解析  焦点电影公司作品,电影焦点结局是什么?  制作网站怎么制作,*游戏网站怎么搭建?  如何快速搭建支持数据库操作的智能建站平台?  如何访问已购建站主机并解决登录问题?  太原网站制作公司有哪些,网约车营运证查询官网?  建站之星在线版空间:自助建站+智能模板一键生成方案  如何通过.red域名打造高辨识度品牌网站?  北京网站制作的公司有哪些,北京白云观官方网站?  如何在IIS中新建站点并配置端口与IP地址?  如何批量查询域名的建站时间记录?  如何用5美元大硬盘VPS安全高效搭建个人网站?  如何解决VPS建站LNMP环境配置常见问题?  国美网站制作流程,国美电器蒸汽鍋怎么用官方网站?  c++怎么用jemalloc c++替换默认内存分配器【性能】  如何高效利用200m空间完成建站?  制作企业网站建设方案,怎样建设一个公司网站? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。