引言

Span<T> 是C# 中的一种结构体,它是一种内存安全的类型,可以用来表示连续的内存区域。Span<T> 可以被用于访问和操作数组、堆上分配的内存和栈上分配的内存。使用 Span<T> 可以避免不必要的内存拷贝,从而提高性能。

对数组使用Span

如果需要快速访问托管或非托管的连续内存,可以使用 Span<T>结构。Span<T> 结构表示存储连续的内存。所以使用它的数据结构一般也使用连续的内存。例如:

  • 数组
  • 长字符串(实际上也是数组)

使用 Span<T>,可以直接访问数组元素。且数组的元素不会复制,可以直接使用它们,这样比复制效率要高。例如下面的代码:

static void Main(string[] args)
{
int[] source = new int[] { 1, 2, 3 }; int[] arr = new int[] { source[0], source[1], source[2] }; arr[0] = 33; Console.WriteLine($"The first element of source is {source[0]}");
Console.WriteLine($"The first element of arr is {arr[0]}"); Span<int> span = new(source); span[0] = 11; Console.WriteLine($"The first element of source is {arr[0]}");
Console.WriteLine($"The first element of span is {span[0]}"); Console.ReadLine();
}

可以先猜测以下上述代码的输出是什么?

输出:

The first element of source is 1
The first element of arr is 33
The first element of source is 11
The first element of span is 11

上述代码段中,先声明了一个源数组 source 和一个数组 arr ,并将 source 的值复制给 arr 。然后修改 arr 中的第一个元素值为33,可以看到结果 arr 的第一个元素已经改变为33,source 保持不变。然后又声明了一个 Span<int> ,它引用 source 数组。因为Span<T>是直接访问数组元素,而不是复制元素,所以修改 span 中的第一个元素为11, source 中的第一个元素也被修改为11。

创建切片

Span<T> 的一个强大特性是,可以使用它访问数组的部分或切片。使用切片时,不会复制数组元素,它们是从span 中直接访问的。

有如下代码段:

static void Main(string[] args)
{
int[] source = { 1, 6, 23, 76, 88, 213 }; Span<int> span1 = new Span<int>(source, start: 1, length: 4); Span<int> span2 = span1.Slice(start: 1, length: 3); DisplaySpan("span1 contains the elements:", span1); DisplaySpan("span2 contains the elements:", span2); Console.ReadLine();
} private static void DisplaySpan(string content, Span<int> span1)
{
Console.Write(content); foreach (var item in span1)
{
Console.Write(item + ",");
} Console.WriteLine();
}

下面的代码片段展示了创建切片的两种方法。

  1. 除默认构造函数传参数组之外,另一种重载是直接使用构造函数传递源数组,起始位置和长度。例如上述代码中 new Span<int>(source, start: 1, length: 4) 它表示在源数组中从第2个元素开始访问数组的4个元素。
  2. 直接从span中再次切片,传入起始位置和长度,例如上述代码中span1.Slice(start: 1, length: 3)表示从span1中第2个元素开始包含3个元素的切片。

输出:

span1 contains the elements:6,23,76,88,
span2 contains the elements:23,76,88,

这里使用时一定注意传入参数 startlength 后的越界问题。

使用Span改变值

在文章开头,介绍了如何使用 Span<T> 的索引器,直接更改由 span 直接引用的数组元素,实际上它还有其他改变值的方法。

例如:

  • Slice(int start, int length):返回一个新的 Span<T>,它表示从 Span<T> 的指定起始位置开始的指定长度部分。可以使用该方法来获取或更改 Span<T> 中的子集。

  • Clear():将 Span<T> 中的所有元素设置为默认值 default<T>

  • Fill(T value):将 Span<T> 中的所有元素设置为指定的值。

  • CopyTo(Span<T> destination):将 Span<T> 中的所有元素复制到指定的目标 Span<T>

  • CopyTo(T[] destination):将 Span<T> 中的所有元素复制到指定的目标数组。

  • Reverse():反转 Span<T> 中的元素顺序。

  • Sort():对 Span<T> 中的元素进行排序。

请注意,这些方法都是按值传递的,而不是按引用传递的。这意味着在调用这些方法时,将复制 Span<T> 中的值。如果您想要修改原始 Span<T> 中的值,请使用引用传递方式,例如使用 ref Span<T> 参数。

只读的Span

如果只需要对数组片段进行读访问,则可以使用 ReadOnlySpan<T>,可以使用它来读取内存块中的数据,而不必担心其他代码同时修改了该内存块。

对于 ReadOnlySpan<T> ,它的索引器是只读的,所以这种类型没有提供 ClearFill 方法,但是可以调用 CopyTo() 方法,将 ReadOnlySpan<T> 的内容复制到 Span<T>

此外,它支持隐式转换,由数组或 Span<T> 直接赋值给 ReadOnlySpan<T>,如下:

static void Main(string[] args)
{
int[] source = { 1, 6, 23, 76, 88, 213 }; Span<int> span = new Span<int>(source); DisplaySpan("span contains the elements:", span); ReadOnlySpan<int> readOnlySpan = source; DisplaySpan("readOnlySpan contains the elements:", readOnlySpan); Console.ReadLine();
} private static void DisplaySpan(string content, ReadOnlySpan<int> span1)
{
Console.Write(content); foreach (var item in span1)
{
Console.Write(item + ",");
} Console.WriteLine();
}

输出:

span contains the elements:1,6,23,76,88,213,
readOnlySpan contains the elements:1,6,23,76,88,213,

Span<T> 相比,ReadOnlySpan<T> 的一个重要的限制是不允许修改其包含的内存块。这使得 ReadOnlySpan<T> 更适合于读取内存块中的数据,而不是修改它们。

解读 --- Span<T>的更多相关文章

  1. C# Span 源码解读和应用实践

    一:背景 1. 讲故事 这两天工作上太忙没有及时持续的文章产出,和大家说声抱歉,前几天群里一个朋友在问什么时候可以产出 Span 的下一篇,哈哈,这就来啦!读过上一篇的朋友应该都知道 Span 统一了 ...

  2. 解读ASP.NET 5 & MVC6系列(13):TagHelper

    在新版的MVC6中,微软提供了强大的TagHelper功能,以便让我们摆脱如下的臃肿代码: @Html.LabelFor(model => model.FullName) @Html.EditF ...

  3. 解读ASP.NET 5 & MVC6系列(1):ASP.NET 5简介

    ASP.NET 5简介 ASP.NET 5是一个跨时代的改写,所有的功能和模块都进行了独立拆分,做到了彻底解耦.为了这些改写,微软也是蛮 拼的,几乎把.NET Framwrok全部改写了一遍,形成了一 ...

  4. 优秀开源代码解读之JS与iOS Native Code互调的优雅实现方案

    简介 本篇为大家介绍一个优秀的开源小项目:WebViewJavascriptBridge. 它优雅地实现了在使用UIWebView时JS与ios 的ObjC nativecode之间的互调,支持消息发 ...

  5. Hybrid----优秀开源代码解读之JS与iOS Native Code互调的优雅实现方案-备

    本篇为大家介绍一个优秀的开源小项目:WebViewJavascriptBridge. 它优雅地实现了在使用UIWebView时JS与ios 的ObjC nativecode之间的互调,支持消息发送.接 ...

  6. Jsoup代码解读之六-防御XSS攻击

    Jsoup代码解读之八-防御XSS攻击 防御XSS攻击的一般原理 cleaner是Jsoup的重要功能之一,我们常用它来进行富文本输入中的XSS防御. 我们知道,XSS攻击的一般方式是,通过在页面输入 ...

  7. Jsoup代码解读之三-Document的输出

    Jsoup代码解读之三-Document的输出   Jsoup官方说明里,一个重要的功能就是output tidy HTML.这里我们看看Jsoup是如何输出HTML的. HTML相关知识 分析代码前 ...

  8. CSS之float属性解读

    在web标准的网页中,页面各个元素都是以标准流的方式来进行布局的.即块元素占满指定的宽度,不指定宽度则占满整行(如<p>.<div>元素),内联元素则是在行内一个接一个的从左到 ...

  9. Angularjs directive全面解读(1.4.5)

    说到Angularjs directive即指令,可以这么说Angularjs的灵魂就是指令,学会Angularjs指令那么你的Angularjs的武功就修炼了一半了,当然这只是鄙人的一点点独到见解, ...

  10. 唱吧DevOps的落地,微服务CI/CD的范本技术解读----最大的难点并不是实际业务代码的编写,而是服务的监控和调试以及容器的编排

    1.业务架构:从单体式到微服务 K歌亭是唱吧的一条新业务线,旨在提供线下便捷的快餐式K歌方式,用户可以在一个电话亭大小的空间里完成K歌体验.K歌亭在客户端有VOD.微信和Web共三个交互入口,业务复杂 ...

随机推荐

  1. Prisim Sample 7 Modules App.Config

    在项目中添加模块化文件.模块文件怎样在主项目中注册.本例 说明方式一,使用了App.config文件. 其中: <?xml version="1.0" encoding=&q ...

  2. 2020-08-27:OpenStack与Docker的区别?

    福哥答案2020-08-27: 此答案来自qq群巨佬:openstack是个体系一个完整的方案系统 基于虚拟机 docker只是一个内核上特殊模式跑与其他系统进程隔离的进程的软件方案真正与docker ...

  3. 2022-10-06:以下go语言代码输出什么?A:[1 2 3] [1 2 3] ;B:[1 2 3] [3 4 5]; C:[1 2 3] [3 4 5 6 7 8 9];D:[1 2 3] [3

    2022-10-06:以下go语言代码输出什么?A:[1 2 3] [1 2 3] :B:[1 2 3] [3 4 5]: C:[1 2 3] [3 4 5 6 7 8 9]:D:[1 2 3] [3 ...

  4. 2022-01-23:力扣425,单词方块。 给定一个单词集合 (没有重复),找出其中所有的 单词方块 。 一个单词序列形成了一个有效的单词方块的意思是指从第 k 行和第 k 列 (0 ≤ k < m

    2022-01-23:力扣425,单词方块. 给定一个单词集合 (没有重复),找出其中所有的 单词方块 . 一个单词序列形成了一个有效的单词方块的意思是指从第 k 行和第 k 列 (0 ≤ k < ...

  5. 2021-07-06:股票问题3。给定一个数组,它的第 i 个元素是一支给定的股票在第 i 天的价格。设计一个算法来计算你所能获取的最大利润。你最多可以完成 两笔 交易。注意:你不能同时参与多笔交易(

    2021-07-06:股票问题3.给定一个数组,它的第 i 个元素是一支给定的股票在第 i 天的价格.设计一个算法来计算你所能获取的最大利润.你最多可以完成 两笔 交易.注意:你不能同时参与多笔交易( ...

  6. java中this的内存原理以及成员变量和局部变量

    this的内存原理 1.this的作用: 区分局部变量和成员变量 eg: public class Student{ private int age; public void method(){ in ...

  7. vue全家桶进阶之路9:常用指令

    以下是一些常见的指令: v-bind - 用于绑定一个或多个属性到组件或 HTML 元素上. v-model - 用于双向绑定一个表单元素或组件的值到数据模型上. v-for - 用于循环遍历一个数组 ...

  8. 记一次,使用python实现一键在爱发电发布带图片的动态

    1.背景 本人喜欢转载一些youtube上的视频到b站上面,然后就会有些观众想要视频的封面,那我总不可能一个一个发吧,太麻烦了.故打算将资源发布到爱发电上面.但是爱发电却没有公开对应的api,只能自己 ...

  9. phpstudy-sqlilabs-less-11

    题目:POST - Error Based - Single quotes- String 基于错误的单引号post型字符变形的注入 看到有个账密输入口第一反应尝试post注入 打开post data ...

  10. 技术招聘漫谈 | 正在招Golang工程师的你,赶快收藏这份识人秘籍!

    各位技术面试官,欢迎来到新一期的技术招聘漫谈专栏. 在前两期的专栏中,我们解析了前端工程师以及 Java 工程师 这两个常见技术岗位的招聘技巧. 今天,我们想把目光聚焦在一个前景与"钱&qu ...