一、前言

其实说到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的性能日后会越来越强大。

相关Demo会上传到QQ群中。

  GitHub:https://github.com/maxzhang1985/YOYOFx 如果觉还可以请Star下, 欢迎一起交流。

  .NET Core 开源学习群:214741894

转载于:https://www.cnblogs.com/maxzhang1985/p/6875622.html

.Net Core中使用ref和SpanT提高程序性能相关推荐

  1. JAVA用多线程反而变慢了_Java中使用多线程不能明显提高程序效率的一些原因

    java中使用多线程不能明显提高程序效率的一些原因. 使用多个线程来处理多任务的时候,效率肯定是有提高的.但是必须要慎用,否则容易出现问题. 1.多线程主要是为了充分利用多核cpu,大内存这些资源. ...

  2. 如何在ASP.NET Core中编写自定义日志记录提供程序

    目录 介绍 如何实现所需的接口 基础类和附件 FileLoggerProvider具体类及其附件 1. ConfigureLogging() 2. appsettings.json文件 介绍 源代码可 ...

  3. [.net 面向对象程序设计进阶] (18) 多线程(Multithreading)(三) 利用多线程提高程序性能(下)...

    [.net 面向对象程序设计进阶] (18) 多线程(Multithreading)(二) 利用多线程提高程序性能(下) 本节导读: 上节说了线程同步中使用线程锁和线程通知的方式来处理资源共享问题,这 ...

  4. 如何使用缓存提高程序性能

    1 写在前面 此文主要参考了园子里以下两篇文章: 黄聪,Microsoft Enterprise Library 5.0 系列(一) : Caching Application Block (初级) ...

  5. .Net Core中使用ref和Spanamp;lt;Tamp;gt;提高程序性能

    一.前言 其实说到ref,很多同学对它已经有所了解,ref是C# 7.0的一个语言特性,它为开发人员提供了返回本地变量引用和值引用的机制. Span 也是建立在ref语法基础上的一个复杂的数据类型,在 ...

  6. 如何在SQL Server 2016中使用并行插入以提高查询性能

    介绍 (Introduction ) In the first part of this article, we will discuss about parallelism in the SQL S ...

  7. Spring MVC 异步处理请求,提高程序性能

    原文:http://blog.csdn.net/he90227/article/details/52262163 什么是异步模式 如何在Spring MVC中使用异步提高性能?一个普通 Servlet ...

  8. 利用多线程提高程序性能(for Android)

    要想搞出一个反应迅速的Android应用程序,一个很好的做法就是确保在主UI线程里执行尽量少的代码.任何有可能花费较长时间来执行的代码如果在主UI线程执行,则会让程序挂起无法响应用户的操作,所以应该放 ...

  9. 如何用AIO技术提高程序性能

    写在前面 这是一篇关于 AIO 的文章.本篇文章详细对比了几个常见的I/O模型,并且介绍了AIO相关的一些API. 我把英文原文翻译过来整理成这篇文章.目的一个是自己学习,一个是方便不习惯看英文资料的 ...

最新文章

  1. Gym 101775J Straight Master(差分数组)题解
  2. Centos下MySql用户管理
  3. 4个常用的awk统计命令
  4. Codevs 2756 树上的路径
  5. python struct pack解析_Python struct 详解
  6. Debian下RPM包安装
  7. Spring : Spring profile 实现多环境支持
  8. 如何使用Dojo的DatePicker控件制作联动日期选择器
  9. idea actiBPM插件生成png文件 (解决没有Diagrams或Designer选项问题)
  10. 关于static 和 const
  11. httpclient案例一(调用识别接口)
  12. flashpaper java_使用FlashPaper 实现JSP在线阅读
  13. 利用js书写正三角形
  14. 给一个大表增加一个字段,带默认值
  15. [市场产品部]MP部副部长(陈晓慧):餐饮公司LOGO设计与制作
  16. Python 常用模块总结
  17. java注册时向指定邮箱发送邮件激活
  18. SRM供应商关系管理系统解决方案
  19. 2021年电工(中级)试题及解析及电工(中级)证考试
  20. Linux 下第一次使用MySQL遇到几种问题。

热门文章

  1. eclipse for php开发环境,eclipse for php 开发环境配置
  2. Gerrit 服务搭建和升级详解(包括 H2 数据库迁移 MySQL 步骤)
  3. geoda空间自相关分析_【方法笔记4】Geoda空间计量1 空间自相关
  4. Intellij IDEA 中的 Debug 控制台输出窗口消失
  5. 攻防世界 WEB 新手练习区 答题(1-12题解)
  6. 【NOIP2015】【Luogu2678】跳石头
  7. mysql锁问题排查_Mysql死锁问题如何排查和解决?
  8. java a =a-=aa_java 初学 :求 s=a+aa+aaa+aaaa+aa...a 的值,其中 a 是一个数字。几个 数相加由键盘控制。...
  9. 软件静态测试qac,Helix QAC — 软件静态测试工具
  10. python将图片转换为字符_python如何将图片转换为字符图片