.Net Core中使用ref和Span<T>提高程序性能的实现代码
发布时间 - 2026-01-11 01:11:46 点击率:次一、前言

其实说到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中的导航属性
# 转换为
# 小编
# 组中
# 都是
# 整型
# 几个
# 我不
# 我会
# 都不
# 也会
# 我觉得
# 如有
# 基础上
# 适用于
# 说到
# 这是一个
# 将是
# 给大家
# 会对
# 早在
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Bootstrap CSS布局之列表
千库网官网入口推荐 千库网设计创意平台入口
Laravel API路由如何设计_Laravel构建RESTful API的路由最佳实践
昵图网官网入口 昵图网素材平台官方入口
深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?
微信h5制作网站有哪些,免费微信H5页面制作工具?
html5如何设置样式_HTML5样式设置方法与CSS应用技巧【教程】
javascript中对象的定义、使用以及对象和原型链操作小结
如何获取上海专业网站定制建站电话?
javascript中数组(Array)对象和字符串(String)对象的常用方法总结
如何在七牛云存储上搭建网站并设置自定义域名?
Laravel如何创建自定义Artisan命令?(代码示例)
猪八戒网站制作视频,开发一个猪八戒网站,大约需要多少?或者自己请程序员,需要什么程序员,多少程序员能完成?
猎豹浏览器开发者工具怎么打开 猎豹浏览器F12调试工具使用【前端必备】
如何生成腾讯云建站专用兑换码?
网站页面设计需要考虑到这些问题
如何在局域网内绑定自建网站域名?
Laravel怎么为数据库表字段添加索引以优化查询
浅谈redis在项目中的应用
Laravel如何监控和管理失败的队列任务_Laravel失败任务处理与监控
如何在IIS中新建站点并解决端口绑定冲突?
如何登录建站主机?访问步骤全解析
电商网站制作价格怎么算,网上拍卖流程以及规则?
Laravel怎么多语言本地化设置_Laravel语言包翻译与Locale动态切换【手册】
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
深圳防火门网站制作公司,深圳中天明防火门怎么编码?
在线ppt制作网站有哪些软件,如何把网页的内容做成ppt?
JS中对数组元素进行增删改移的方法总结
Laravel如何实现多对多模型关联?(Eloquent教程)
如何在 React 中条件性地遍历数组并渲染元素
Laravel怎么使用Intervention Image库处理图片上传和缩放
惠州网站建设制作推广,惠州市华视达文化传媒有限公司怎么样?
实例解析Array和String方法
想要更高端的建设网站,这些原则一定要坚持!
Laravel中的Facade(门面)到底是什么原理
Laravel如何配置和使用队列处理异步任务_Laravel队列驱动与任务分发实例
EditPlus中的正则表达式 实战(4)
敲碗10年!Mac系列传将迎来「触控与联网」双革新
如何在云主机上快速搭建网站?
如何制作一个表白网站视频,关于勇敢表白的小标题?
Android滚轮选择时间控件使用详解
Python进程池调度策略_任务分发说明【指导】
Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】
JavaScript常见的五种数组去重的方式
Android 常见的图片加载框架详细介绍
php打包exe后无法访问网络共享_共享权限设置方法【教程】
微信推文制作网站有哪些,怎么做微信推文,急?
香港服务器网站生成指南:免费资源整合与高速稳定配置方案
Win11怎么恢复误删照片_Win11数据恢复工具使用【推荐】
Laravel怎么在Controller之外的地方验证数据

