主頁 > .NET開發 > C# 11 對 ref 和 struct 的改進

C# 11 對 ref 和 struct 的改進

2022-04-22 06:05:53 .NET開發

前言

C# 11 中即將到來一個可以讓重視性能的開發者狂喜的重量級特性,這個特性主要是圍繞著一個重要底層性能設施 refstruct 的一系列改進,

但是這部分的改進涉及的內容較多,不一定能在 .NET 7(C# 11)做完,因此部分內容推遲到 C# 12 也是有可能的,當然,還是很有希望能在 C# 11 的時間點就看到完全體的,

本文僅僅就這一個特性進行介紹,因為 C# 11 除了本特性之外,還有很多其他的改進,一篇文章根本說不完,其他那些我們就等到 .NET 7 快正式發布的時候再說吧,

背景

C# 自 7.0 版本引入了新的 ref struct 用來表示不可被裝箱的堆疊上物件,但是當時局限性很大,甚至無法被用于泛型約束,也無法作為 struct 的欄位,在 C# 11 中,由于特性 ref 欄位的推動,需要允許型別持有其它值型別的參考,這方面的東西終于有了大幅度進展,

這些設施旨在允許開發者使用安全的代碼撰寫高性能代碼,而無需面對不安全的指標,接下來我就來對 C# 11 甚至 12 在此方面的即將到來的改進進行介紹,

ref 欄位

C# 以前是不能在型別中持有對其它值型別的參考的,但是在 C# 11 中,這將變得可能,從 C# 11 開始,將允許 ref struct 定義 ref 欄位,

readonly ref struct Span<T>
{
    private readonly ref T _field;
    private readonly int _length;
    public Span(ref T value)
    {
        _field = ref value;
        _length = 1;
    }
}

直觀來看,這樣的特性將允許我們寫出上面的代碼,這段代碼中構造了一個 Span<T>,它持有了對其他 T 物件的參考,

當然,ref struct 也是可以被 default 來初始化的:

Span<int> span = default;

但這樣 _field 就會是個空參考,不過我們可以通過 Unsafe.IsNullRef 方法來進行檢查:

if (Unsafe.IsNullRef(ref _field))
{
    throw new NullReferenceException(...);
}

另外,ref欄位的可修改性也是一個非常重要的事情,因此引入了:

  • readonly ref:一個對物件的只讀參考,這個參考本身不能在構造方法或 init 方法之外被修改
  • ref readonly:一個對只讀物件的參考,這個參考指向的物件不能在構造方法或 init 方法之外被修改
  • readonly ref readonly:一個對只讀物件的只讀參考,是上述兩種的組合

例如:

ref struct Foo
{
    ref readonly int f1;
    readonly ref int f2;
    readonly ref readonly int f3;

    void Bar(int[] array)
    {
        f1 = ref array[0];  // 沒問題
        f1 = array[0];      // 錯誤,因為 f1 參考的值不能被修改
        f2 = ref array[0];  // 錯誤,因為 f2 本身不能被修改
        f2 = array[0];      // 沒問題
        f3 = ref array[0];  // 錯誤:因為 f3 本身不能被修改
        f3 = array[0];      // 錯誤:因為 f3 參考的值不能被修改
    }
}

生命周期

這一切看上去都很美好,但是真的沒有任何問題嗎?

假設我們有下面的代碼來使用上面的東西:

Span<int> Foo()
{
    int v = 42;
    return new Span<int>(ref v);
}

v 是一個區域變數,在函式回傳之后其生命周期就會結束,那么上面這段代碼就會導致 Span<int> 持有的 v 的參考變成無效的,順帶一提,上面這段代碼是完全合法的,因為 C# 之前不支持 ref 欄位,因此上面的代碼是不可能出現逃逸問題的,但是 C# 11 加入了 ref 欄位,堆疊上的物件就有可能通過 ref 欄位而發生參考逃逸,于是代碼變得不安全,

如果我們有一個 CreateSpan 方法用來創建一個參考的 Span

Span<int> CreateSpan(ref int v)
{
     // ...
}

這就衍生出了一系列在以前的 C# 中沒問題(因為 ref 的生命周期為當前方法),但是在 C# 11 中由于可能存在 ref 欄位而導致用安全的方式寫出的非安全代碼:

Span<int> Foo(int v)
{
    // 1
    return CreateSpan(ref v);

    // 2
    int local = 42;
    return CreateSpan(ref local);

    // 3
    Span<int> span = stackalloc int[42];
    return CreateSpan(ref span[0]);
}

因此,在 C# 11 中則不得不引入破壞性更改,不允許上述代碼通過編譯,但這并沒有完全解決問題,

為了解決逃逸問題, C# 11 制定了參考逃逸安全規則,對于一個在 e 中的欄位 f

  • 如果 f 是個 ref 欄位,并且 ethis,則 f 在它被包圍的方法中是參考逃逸安全的
  • 否則如果 f 是個 ref 欄位,則 f 的參考逃逸安全范圍和 e 的逃逸安全范圍相同
  • 否則如果 e 是一個參考型別,則 f 的參考逃逸安全范圍是呼叫它的方法
  • 否則 f 的參考逃逸安全范圍和 e 相同

由于 C# 中的方法是可以回傳參考的,因此根據上面的規則,一個 ref struct 中的方法將不能回傳一個對非 ref 欄位的參考:

ref struct Foo
{
    private ref int _f1;
    private int f2;

    public ref int P1 => ref _f1; // 沒問題
    public ref int P2 => ref _f2; // 錯誤,因為違反了第四條規則
}

除了參考逃逸安全規則之外,同樣還有對 ref 賦值的規則:

  • 對于 x.e1 = ref e2, 其中 x 是在呼叫方法中逃逸安全的,那么 e2 必須在呼叫方法中是參考逃逸安全的
  • 對于 e1 = ref e2,其中 e1 是個區域變數,那么 e2 的參考逃逸安全范圍必須至少和 e1 的參考逃逸安全范圍一樣大

于是, 根據上述規則,下面的代碼是沒問題的:

readonly ref struct Span<T>
{
    readonly ref T _field;
    readonly int _length;

    public Span(ref T value)
    {
        // 沒問題,因為 x 是 this,this 的逃逸安全范圍和 value 的參考逃逸安全范圍都是呼叫方法,滿足規則 1
        _field = ref value;
        _length = 1;
    }
}

于是很自然的,就需要在欄位和引數上對生命周期進行標注,幫助編譯器確定物件的逃逸范圍,

而我們在寫代碼的時候,并不需要記住以上這么多的規則,因為有了生命周期標注之后一切都變得顯式和直觀了,

scoped

在 C# 11 中,引入了 scoped 關鍵字用來限制逃逸安全范圍:

區域變數 s 參考逃逸安全范圍 逃逸安全范圍
Span<int> s 當前方法 呼叫方法
scoped Span<int> s 當前方法 當前方法
ref Span<int> s 呼叫方法 呼叫方法
scoped ref Span<int> s 當前方法 呼叫方法
ref scoped Span<int> s 當前方法 當前方法
scoped ref scoped Span<int> s 當前方法 當前方法

其中,scoped ref scoped 是多余的,因為它可以被 ref scoped 隱含,而我們只需要知道 scoped 是用來把逃逸范圍限制到當前方法的即可,是不是非常簡單?

如此一來,我們就可以對引數進行逃逸范圍(生命周期)的標注:

Span<int> CreateSpan(scoped ref int v)
{
    // ...
}

然后,之前的代碼就變得沒問題了,因為都是 scoped ref

Span<int> Foo(int v)
{
    // 1
    return CreateSpan(ref v);

    // 2
    int local = 42;
    return CreateSpan(ref local);

    // 3
    Span<int> span = stackalloc int[42];
    return CreateSpan(ref span[0]);
}

scoped 同樣可以被用在區域變數上:

Span<int> Foo()
{
    // 錯誤,因為 span 不能逃逸當前方法
    scoped Span<int> span1 = default;
    return span1;

    // 沒問題,因為初始化器的逃逸安全范圍是呼叫方法,因為 span2 可以逃逸到呼叫方法
    Span<int> span2 = default;
    return span2;

    // span3 和 span4 是一樣的,因為初始化器的逃逸安全范圍是當前方法,加不加 scoped 都沒區別
    Span<int> span3 = stackalloc int[42];
    scoped Span<int> span4 = stackalloc int[42];
}

另外,structthis 也加上了 scoped ref 的逃逸范圍,即參考逃逸安全范圍為當前方法,而逃逸安全范圍為呼叫方法,

剩下的就是和 outin 引數的配合,在 C# 11 中,out 引數將會默認為 scoped ref,而 in 引數仍然保持默認為 ref

ref int Foo(out int r)
{
    r = 42;
    return ref r; // 錯誤,因為 r 的參考逃逸安全范圍是當前方法
}

這非常有用,例如比如下面這個常見的情況:

Span<byte> Read(Span<byte> buffer, out int read)
{
    // .. 
}

Span<int> Use()
{
    var buffer = new byte[256];

    // 如果不修改 out 的參考逃逸安全范圍,則這會報錯,因為編譯器需要考慮 read 是可以被作為 ref 欄位回傳的情況
    // 如果修改 out 的參考逃逸安全范圍,則就沒有問題了,因為編譯器不需要考慮 read 是可以被作為 ref 欄位回傳的情況
    int read;
    return Read(buffer, out read);
}

下面給出一些更多的例子:

Span<int> CreateWithoutCapture(scoped ref int value)
{
    // 錯誤,因為 value 的參考逃逸安全范圍是當前方法
    return new Span<int>(ref value);
}

Span<int> CreateAndCapture(ref int value)
{
    // 沒問題,因為 value 的逃逸安全范圍被限制為 value 的參考逃逸安全范圍,這個范圍是呼叫方法
    return new Span<int>(ref value)
}

Span<int> ComplexScopedRefExample(scoped ref Span<int> span)
{
    // 沒問題,因為 span 的逃逸安全范圍是呼叫方法
    return span;

    // 沒問題,因為 refLocal 的參考逃逸安全范圍是當前方法、逃逸安全范圍是呼叫方法
    // 在 ComplexScopedRefExample 的呼叫中它被傳遞給了一個 scoped ref 引數,
    // 意味著編譯器在計算生命周期時不需要考慮參考逃逸安全范圍,只需要考慮逃逸安全范圍
    // 因此它回傳的值的安全逃逸范圍為呼叫方法
    Span<int> local = default;
    ref Span<int> refLocal = ref local;
    return ComplexScopedRefExample(ref refLocal);

    // 錯誤,因為 stackLocal 的參考逃逸安全范圍、逃逸安全范圍都是當前方法
    // 在 ComplexScopedRefExample 的呼叫中它被傳遞給了一個 scoped ref 引數,
    // 意味著編譯器在計算生命周期時不需要考慮參考逃逸安全范圍,只需要考慮逃逸安全范圍
    // 因此它回傳的值的安全逃逸范圍為當前方法
    Span<int> stackLocal = stackalloc int[42];
    return ComplexScopedRefExample(ref stackLocal);
}

unscoped

上述的設計中,仍然有個問題沒有被解決:

struct S
{
    int _field;

    // 錯誤,因為 this 的參考逃逸安全范圍是當前方法
    public ref int Prop => ref _field;
}

因此引入一個 unscoped,允許擴展逃逸范圍到呼叫方法上,于是,上面的方法可以改寫為:

struct S
{
    private int _field;

    // 沒問題,參考逃逸安全范圍被擴展到了呼叫方法
    public unscoped ref int Prop => ref _field;
}

這個 unscoped 也可以直接放到 struct 上:

unscoped struct S
{
    private int _field;
    public unscoped ref int Prop => ref _field;
}

同理,嵌套的 struct 也沒有問題:

unscoped struct Child
{
    int _value;
    public ref int Value =https://www.cnblogs.com/hez2010/p/> ref _value;
}

unscoped struct Container
{
    Child _child;
    public ref int Value => ref _child.Value;
}

此外,如果需要恢復以前的 out 逃逸范圍的話,也可以在 out 引數上指定 unscoped

ref int Foo(unscoped out int r)
{
    r = 42;
    return ref r;
}

不過有關 unscoped 的設計還屬于初步階段,不會在 C# 11 中就提供,

ref struct 約束

從 C# 11 開始,ref struct 可以作為泛型約束了,因此可以撰寫如下方法了:

void Foo<T>(T v) where T : ref struct
{
    // ...
}

因此,Span<T> 的功能也被擴展,可以宣告 Span<Span<T>> 了,比如用在 byte 或者 char 上,就可以用來做高性能的字串處理了,

反射

有了上面那么多東西,反射自然也是要支持的,因此,反射 API 也加入了 ref struct 相關的支持,

實際用例

有了以上基礎設施之后,我們就可以使用安全代碼來造一些高性能輪子了,

堆疊上定長串列

struct FrugalList<T>
{
    private T _item0;
    private T _item1;
    private T _item2;

    public readonly int Count = 3;

    public unscoped ref T this[int index] => index switch
    {
        0 => ref _item1,
        1 => ref _item2,
        2 => ref _item3,
        _ => throw new OutOfRangeException("Out of range.")
    };
}

堆疊上鏈表

ref struct StackLinkedListNode<T>
{
    private T _value;
    private ref StackLinkedListNode<T> _next;

    public T Value =https://www.cnblogs.com/hez2010/p/> _value;
    public bool HasNext => !Unsafe.IsNullRef(ref _next);
    public ref StackLinkedListNode Next => HasNext ? ref _next : throw new InvalidOperationException("No next node.");

    public StackLinkedListNode(T value)
    {
        this = default;
        _value = https://www.cnblogs.com/hez2010/p/value;
    }

    public StackLinkedListNode(T value, ref StackLinkedListNode next)
    {
        _value = value;
        _next = ref next;
    }
}

除了這兩個例子之外,其他的比如決議器和序列化器等等,例如 Utf8JsonReaderUtf8JsonWriter 都可以用到這些東西,

未來計劃

高級生命周期

上面的生命周期設計雖然能滿足絕大多數使用,但是還是不夠靈活,因此未來有可能在此基礎上擴展,引入高級生命周期標注,例如:

void M(scoped<'a> ref MyStruct s, scoped<'b> Span<int> span) where 'b >= 'a
{
    s.Span = span;
}

上面的方法給引數 sspan 分別宣告了兩個生命周期 'a'b,并約束 'b 的生命周期不小于 'a,因此在這個方法里,span 可以安全地被賦值給 s.Span

這個雖然不會被包含在 C# 11 中,但是如果以后開發者對相關的需求增長,是有可能被后續加入到 C# 中的,

總結

以上就是 C# 11(或之后)對 refstruct 的改進了,有了這些基礎設施,開發者們將能輕松使用安全的方式來撰寫沒有任何堆記憶體開銷的高性能代碼,盡管這些改進只能直接讓小部分非常關注性能的開發者收益,但是這些改進帶來的將是后續基礎庫代碼質量和性能的整體提升,

如果你擔心這會讓語言的復雜度上升,那也大可不必,因為這些東西大多數人并不會用到,只會影響到小部分的開發者,因此對于大多數人而言,只需要寫著原樣的代碼,享受其他基礎庫作者利用上述設施撰寫好的東西即可,

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

標籤:C#

上一篇:09. 樹莓派ASP.NET環境配置

下一篇:ROZRZ在線測量自動測量在線刀補CNC遠程刀補機床遠程刀聯對刀儀遠程刀補掃碼傳輸刀補

標籤雲
其他(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