主頁 > .NET開發 > 關于C# Span的一些實踐

關于C# Span的一些實踐

2020-11-25 09:26:35 .NET開發

Span這個東西出來很久了,居然因為5.0又火起來了,

?

相關知識

在大多數情況下,C#開發時,我們只使用托管記憶體,而實際上,C#為我們提供了三種型別的記憶體:

  • 堆疊記憶體 - 最快速的記憶體,能夠做到極快的分配和釋放,堆疊記憶體使用時,需要用stackalloc進行分配,堆疊的一個特點是空間非常小(通常小于1 MB),適合CPU快取,試圖分配更多堆疊會報出StackOverflowException錯誤并終止行程;另一個特點是生命周期非常短 - 方法結束時,堆疊會與方法的記憶體一起釋放,stackalloc通常用于必須不分配任何托管記憶體的短操作,一個例子是在corefx中記錄快速記錄ETW事件:要求盡可能快,并且需要很少的記憶體,
  • 非托管記憶體 - 通過Marshal.AllocHGlobalxMarshal.AllocCoTaskMem方法分配在非托管堆上的記憶體,這個記憶體對GC不可見,并且必須通過Marshal.FreeHGlobalMarshal.FreeCoTaskMem的顯式呼叫來釋放,使用非托管記憶體,最主要的目的是不給GC增加額外的壓力,所以最經常的使用方式是在分配大量沒有指標的值型別時使用,在Kestrel的代碼中,很多地方用到了非托管記憶體,
  • 托管記憶體 - 大多數代碼中最常用的記憶體,需要用new運算子來分配,之所以稱為托管(managed),因為它是被GC(垃圾管理器)管理的,由GC決定何時釋放記憶體,而不需要開發人員考慮,GC又將托管物件根據大小(85000位元組)分為大物件和小物件,兩個物件的分配方式、速度和位置都有不同,小物件相對快點,大物件相對慢點,另外,兩種物件的GC回收成本也不一樣,

    為防止非授權轉發,這兒給出本文的原文鏈接:https://www.cnblogs.com/tiger-wang/p/14029853.html

問題的產生

問個問題:寫了這么多年的C#,我們有用過指標嗎?有沒有想過為什么?

我們用個例子來回答這個問題:一個字串,正常它是一個托管物件,

如果我們想決議整個字串,我們會這么寫:

int Parse(string managedMemory);

那么,如果我們想只決議一部分字串,該怎么寫?

int Parse(string managedMemory, int startIndex, int length);

現在,我們轉到非托管記憶體上:

unsafe int Parse(char* pointerToUnmanagedMemory, int length);
unsafe int Parse(char* pointerToUnmanagedMemory, int startIndex, int length);

再延伸一下,我們寫幾個用于復制記憶體的功能:

void Copy<T>(T[] source, T[] destination); 
void Copy<T>(T[] source, int sourceStartIndex, T[] destination, int destinationStartIndex, int elementsCount);
unsafe void Copy<T>(void* source, void* destination, int elementsCount);
unsafe void Copy<T>(void* source, int sourceStartIndex, void* destination, int destinationStartIndex, int elementsCount);
unsafe void Copy<T>(void* source, int sourceLength, T[] destination);
unsafe void Copy<T>(void* source, int sourceStartIndex, T[] destination, int destinationStartIndex, int elementsCount);

是不是很復雜?而且看上去并不安全?

所以,問題并不在于我們能不能用,而在于這種支持會讓代碼變得復雜,而且并不安全 - 直到Span出現,

Span

在定義中,Span就是一個簡單的值型別,它真正的價值,在于允許我們與任何型別的連續記憶體一起作業,

這些所謂的連續記憶體,包括:

  • 非托管記憶體緩沖區
  • 陣列和子串
  • 字串和子字串

在使用中,Span確保了記憶體和資料安全,而且幾乎沒有開銷,

使用Span

要使用Span,需要設定開發語言為C# 7.2以上,并參考System.Memory到專案,

<PropertyGroup>
  <LangVersion>7.2</LangVersion>
</PropertyGroup>

使用低版本編譯器,會報錯:Error CS8107 Feature 'ref structs' is not available in C# 7.0. Please use language version 7.2 or greater.

?

Span使用時,最簡單的,可以把它想象成一個陣列,它會做所有的指標運算,同時,內部又可以指向任何型別的記憶體,

例如,我們可以為非托管記憶體創建Span:

Span<byte> stackMemory = stackalloc byte[256];

IntPtr unmanagedHandle = Marshal.AllocHGlobal(256);
Span<byte> unmanaged = new Span<byte>(unmanagedHandle.ToPointer(), 256); 
Marshal.FreeHGlobal(unmanagedHandle);

T[]到Span的隱式轉換:

char[] array = new char[] { 'i''m''p''l''i''c''i''t' };
Span<char> fromArray = array;

?

此外,還有ReadOnlySpan,可以用來處理字串或其他不可變型別:

ReadOnlySpan<char> fromString = "Hello world".AsSpan();

?

Span創建完成后,就跟普通的陣列一樣,有一個Length屬性和一個允許讀寫的index,因此使用時就和一般的陣列一樣使用就好,

看看Span常用的一些定義、屬性和方法:

Span(T[] array);
Span(T[] arrayint startIndex);
Span(T[] arrayint startIndex, int length);
unsafe Span(void* memory, int length);

int Length { get; }
ref T this[int index] { get; set; }

Span<T> Slice(int start);
Span<T> Slice(int start, int length);

void Clear();
void Fill(T value);

void CopyTo(Span<T> destination);
bool TryCopyTo(Span<T> destination);

?

我們用Span來實作一下文章開頭的復制記憶體的功能:

int Parse(ReadOnlySpan<char> anyMemory);
int Copy<T>(ReadOnlySpan<T> source, Span<T> destination);

看看,是不是非常簡單?

而且,使用Span時,運行性能極佳,關于Span的性能,網上有很多評測,關注的兄弟可以自己去看,

Span的限制

Span支持所有型別的記憶體,所以,它也會有相當嚴格的限制,

在上面的例子中,使用的是堆疊記憶體,所有指向堆疊的指標都不能存盤在托管堆上,因為方法結束時,堆疊會被釋放,指標會變成無效值,如果再使用,就是記憶體溢位,

因此:Span實體也不能駐留在托管堆上,而只能駐留在堆疊上,這又引出一些限制,

  1. Span不能是非堆疊型別的欄位

如果在類中設定Span欄位,它將被存盤在堆中,這是不允許的:

class Impossible
{

    Span<byte> field;
}

不過,從C# 7.2開始,在其他僅限堆疊的型別中有Span欄位是可以的:

ref struct TwoSpans<T>
{

    public Span<T> first;
    public Span<T> second;

  1. Span不能有介面實作

介面實作意味著資料會被裝箱,而裝箱意味著存盤在堆中,同時,為了防止裝箱,Span必須不實作任何現有的介面,例如最容易想到的IEnumerable,也許某一天,C#會允許定義由結構體實作的結口?

  1. Span不能是異步方法的引數

異步在C#里絕對是個好東西,

不過對于Span,是另一件事,異步方法會創建一個AsyncMethodBuilder構建器,構建器會創建一個異步狀態機,異步狀態機會將方法的引數放到堆上,所以,Span不能用作異步方法的引數,

  1. Span不能是泛型的代入引數

看下面的代碼:

Span<byte> Allocate() => new Span<byte>(new byte[256]);

void CallAndPrint<T>(Func<T> valueProvider) 
{
    object value = valueProvider.Invoke();

    Console.WriteLine(value.ToString());
}

void Demo()
{
    Func<Span<byte>> spanProvider = Allocate;
    CallAndPrint<Span<byte>>(spanProvider);
}

同樣也是裝箱的原因,

?

上面是Span的內容,

下面簡單說一下另一個經常跟Span一起提的內容:Memory

Memory

Memory是一個新的資料型別,它只能指向托管記憶體,所以不具有僅限堆疊的限制,

Memory可以從托管陣列、字串或IOwnedMemory中創建,傳遞給異步方法或存盤在類的欄位中,當需要Span時,就呼叫它的Span屬性,它會根據需要創建Span,然后在當前范圍內使用它,

看一下Memory的主要定義、屬性和方法:

public readonly struct Memory<T>
{

    private readonly object _object;
    private readonly int _index;
    private readonly int _length;

    public Span<T> Span { get; }

    public Memory<T> Slice(int start)
    public Memory<T> Slice(int start, int length)
    public MemoryHandle Pin()
}

使用也很簡單:

byte[] buffer = ArrayPool<byte>.Shared.Rent(16000 * 8);

while ((bytesRead = await fileStream.ReadAsync(buffer, 0, buffer.Length)) > 0)
{
    ParseBlock(new ReadOnlyMemory<byte>(buffer, start: 0, length: bytesRead)); 
}

void ParseBlock(ReadOnlyMemory<byte> memory)
{
    ReadOnlySpan<byte> slice = memory.Span;
}

總結

Span存在很長時間了,只是5.0做了一些優化,

用好了,對代碼是很好的補充和優化,用不好,就會有給自己刨很多個坑,

所以,耗子尾汁,

 


 

微信公眾號:老王Plus

掃描二維碼,關注個人公眾號,可以第一時間得到最新的個人文章和內容推送

本文著作權歸作者所有,轉載請保留此宣告和原文鏈接

轉載請註明出處,本文鏈接:https://www.uj5u.com/net/227459.html

標籤:C#

上一篇:selenium元素定位問題

下一篇:關于C# Span的一些實踐

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • WebAPI簡介

    Web體系結構: 有三個核心:資源(resource),URL(統一資源識別符號)和表示 他們的關系是這樣的:一個資源由一個URL進行標識,HTTP客戶端使用URL定位資源,表示是從資源回傳資料,媒體型別是資源回傳的資料格式。 接下來我們說下HTTP. HTTP協議的系統是一種無狀態的方式,使用請求/ ......

    uj5u.com 2020-09-09 22:07:47 more
  • asp.net core 3.1 入口:Program.cs中的Main函式

    本文分析Program.cs 中Main()函式中代碼的運行順序分析asp.net core程式的啟動,重點不是剖析原始碼,而是理清程式開始時執行的順序。到呼叫了哪些實體,哪些法方。asp.net core 3.1 的程式入口在專案Program.cs檔案里,如下。ususing System; us ......

    uj5u.com 2020-09-09 22:07:49 more
  • asp.net網站作為websocket服務端的應用該如何寫

    最近被websocket的一個問題困擾了很久,有一個需求是在web網站中搭建websocket服務。客戶端通過網頁與服務器建立連接,然后服務器根據ip給客戶端網頁發送資訊。 其實,這個需求并不難,只是剛開始對websocket的內容不太了解。上網搜索了一下,有通過asp.net core 實作的、有 ......

    uj5u.com 2020-09-09 22:08:02 more
  • ASP.NET 開源匯入匯出庫Magicodes.IE Docker中使用

    Magicodes.IE在Docker中使用 更新歷史 2019.02.13 【Nuget】版本更新到2.0.2 【匯入】修復單列匯入的Bug,單元測驗“OneColumnImporter_Test”。問題見(https://github.com/dotnetcore/Magicodes.IE/is ......

    uj5u.com 2020-09-09 22:08:05 more
  • 在webform中使用ajax

    如果你用過Asp.net webform, 說明你也算是.NET 開發的老兵了。WEBform應該是2011 2013左右,當時還用visual studio 2005、 visual studio 2008。后來基本都用的是MVC。 如果是新開發的專案,估計沒人會用webform技術。但是有些舊版 ......

    uj5u.com 2020-09-09 22:08:50 more
  • iis添加asp.net網站,訪問提示:由于擴展配置問題而無法提供您請求的

    今天在iis服務器配置asp.net網站,遇到一個問題,記錄一下: 問題:由于擴展配置問題而無法提供您請求的頁面。如果該頁面是腳本,請添加處理程式。如果應下載檔案,請添加 MIME 映射。 WindowServer2012服務器,添加角色安裝完.netframework和iis之后,運行aspx頁面 ......

    uj5u.com 2020-09-09 22:10:00 more
  • WebAPI-處理架構

    帶著問題去思考,大家好! 問題1:HTTP請求和回傳相應的HTTP回應資訊之間發生了什么? 1:首先是最底層,托管層,位于WebAPI和底層HTTP堆疊之間 2:其次是 訊息處理程式管道層,這里比如日志和快取。OWIN的參考是將訊息處理程式管道的一些功能下移到堆疊下端的OWIN中間件了。 3:控制器處理 ......

    uj5u.com 2020-09-09 22:11:13 more
  • 微信門戶開發框架-使用指導說明書

    微信門戶應用管理系統,采用基于 MVC + Bootstrap + Ajax + Enterprise Library的技術路線,界面層采用Boostrap + Metronic組合的前端框架,資料訪問層支持Oracle、SQLServer、MySQL、PostgreSQL等資料庫。框架以MVC5,... ......

    uj5u.com 2020-09-09 22:15:18 more
  • WebAPI-HTTP編程模型

    帶著問題去思考,大家好!它是什么?它包含什么?它能干什么? 訊息 HTTP編程模型的核心就是訊息抽象,表示為:HttPRequestMessage,HttpResponseMessage.用于客戶端和服務端之間交換請求和回應訊息。 HttpMethod類包含了一組靜態屬性: private stat ......

    uj5u.com 2020-09-09 22:15:23 more
  • 部署WebApi隨筆

    一、跨域 NuGet參考Microsoft.AspNet.WebApi.Cors WebApiConfig.cs中配置: // Web API 配置和服務 config.EnableCors(new EnableCorsAttribute("*", "*", "*")); 二、清除默認回傳XML格式 ......

    uj5u.com 2020-09-09 22:15:48 more
最新发布
  • C#多執行緒學習(二) 如何操縱一個執行緒

    <a href="https://www.cnblogs.com/x-zhi/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2943582/20220801082530.png" alt="" /></...

    uj5u.com 2023-04-19 09:17:20 more
  • C#多執行緒學習(二) 如何操縱一個執行緒

    C#多執行緒學習(二) 如何操縱一個執行緒 執行緒學習第一篇:C#多執行緒學習(一) 多執行緒的相關概念 下面我們就動手來創建一個執行緒,使用Thread類創建執行緒時,只需提供執行緒入口即可。(執行緒入口使程式知道該讓這個執行緒干什么事) 在C#中,執行緒入口是通過ThreadStart代理(delegate)來提供的 ......

    uj5u.com 2023-04-19 09:16:49 more
  • 記一次 .NET某醫療器械清洗系統 卡死分析

    <a href="https://www.cnblogs.com/huangxincheng/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/214741/20200614104537.png" alt="" /&g...

    uj5u.com 2023-04-18 08:39:04 more
  • 記一次 .NET某醫療器械清洗系統 卡死分析

    一:背景 1. 講故事 前段時間協助訓練營里的一位朋友分析了一個程式卡死的問題,回過頭來看這個案例比較經典,這篇稍微整理一下供后來者少踩坑吧。 二:WinDbg 分析 1. 為什么會卡死 因為是表單程式,理所當然就是看主執行緒此時正在做什么? 可以用 ~0s ; k 看一下便知。 0:000> k # ......

    uj5u.com 2023-04-18 08:33:10 more
  • SignalR, No Connection with that ID,IIS

    <a href="https://www.cnblogs.com/smartstar/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/u36196.jpg" alt="" /></a>...

    uj5u.com 2023-03-30 17:21:52 more
  • 一次對pool的誤用導致的.net頻繁gc的診斷分析

    <a href="https://www.cnblogs.com/dotnet-diagnostic/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/3115652/20230225090434.png" alt=""...

    uj5u.com 2023-03-28 10:15:33 more
  • 一次對pool的誤用導致的.net頻繁gc的診斷分析

    <a href="https://www.cnblogs.com/dotnet-diagnostic/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/3115652/20230225090434.png" alt=""...

    uj5u.com 2023-03-28 10:13:31 more
  • C#遍歷指定檔案夾中所有檔案的3種方法

    <a href="https://www.cnblogs.com/xbhp/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/957602/20230310105611.png" alt="" /></a&...

    uj5u.com 2023-03-27 14:46:55 more
  • C#/VB.NET:如何將PDF轉為PDF/A

    <a href="https://www.cnblogs.com/Carina-baby/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2859233/20220427162558.png" alt="" />...

    uj5u.com 2023-03-27 14:46:35 more
  • 武裝你的WEBAPI-OData聚合查詢

    <a href="https://www.cnblogs.com/podolski/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/616093/20140323000327.png" alt="" /><...

    uj5u.com 2023-03-27 14:46:16 more