[譯]C# 7系列,Part 7: ref Returns ref返回結果

来源:https://www.cnblogs.com/wenhx/archive/2019/12/13/csharp-7-series-part-7-ref-returns.html
-Advertisement-
Play Games

原文:https://blogs.msdn.microsoft.com/mazhou/2017/12/12/c-7-series-part-7-ref-returns/ 背景 有兩種方法可以將一個值傳遞給一個方法: 例如,FCL(.NET Framework Class Library)中的Arra ...


原文:https://blogs.msdn.microsoft.com/mazhou/2017/12/12/c-7-series-part-7-ref-returns/

背景

有兩種方法可以將一個值傳遞給一個方法:

  1. 按值傳遞。當一個參數被傳遞給一個方法時,一個參數的副本(如果它是一個值類型)或一個"參數引用"的副本(如果它是一個引用類型)被傳遞。當您更改方法中的參數時,更改(單個賦值或複合賦值)會反映到參數/"參數引用"的副本,而不會反映到參數或“參數引用”本身。這是.NET語言中的預設方式(Visual Basic中的ByVal)。(譯註:"參數引用"其實是個指針,指向另外一個它實際代表的對象,這裡指的是修改這個指針本身。)
  2. 以引用的方式傳遞。當一個參數被傳遞給一個方法時,要麼參數本身(如果它是一個值類型)要麼“參數引用”(如果它是一個引用類型)被直接傳遞。沒有生成其他副本。當您更改方法中的參數時,更改(單個賦值或複合賦值)將反映到參數或“參數引用”本身。這可以通過在C#中使用ref關鍵字或者在Visual Basic中使用ByRef關鍵字來表明。

例如,FCL(.NET Framework Class Library)中的Array. resize()方法接受類型為Array的ref參數,該ref值在方法實現中進行了修改,以指向調整大小後的數組新空間。你可以繼續使用該數組變數,並訪問新的記憶體空間:

byte[] data = new byte[10];
Array.Resize(ref data, 20); //譯註:上一行data指向的記憶體空間和這裡data指向的記憶體空間可能變更了。
Console.WriteLine($"New array size is: {data.Length}");

ref返回結果

ref返回結果是方法的引用返回值。與作為方法參數傳遞的ref值類似,調用者可以修改ref返回值,對該返回值的任何更改(賦值)都將反映到從方法中返回的原始值。

在C#中,你可以使用return ref關鍵字創建一個ref返回結果。

請看下麵的示例:

private static ref T ElementAt<T>(ref T[] array, int position)
{
     if (array == null)
     {
         throw new ArgumentNullException(nameof(array));
     }

     if (position < 0 || position >= array.Length)
     {
         throw new ArgumentOutOfRangeException(nameof(position));
     }

     return ref array[position];
}

上述方法的目的是在數組的特定位置獲取對元素的引用。稍後你可以使用這個引用來更改該元素的值;因為它是一個ref值,所以更改將應用於數組中的原始值。(譯註:就是數組中的原始值會被改變)

要使用這個方法,需要使用ref局部變數:

private static void Main(string[] args)
{
    int[] data = new int[10];
    Console.WriteLine($"Before change, element at 2 is: {data[2]}");
    ref int value = ref ElementAt(ref data, 2);
    // Change the ref value.
    value = 5;
    Console.WriteLine($"After change, element at 2 is: {data[2]}");
}

Visual Studio智能感知錶面調用的方法是一個ref返回結果方法。

上面代碼的運行輸出結果如下:

調用帶有ref返回結果的方法

與前面的示例一樣,要獲得ref返回值的引用,你需要使用ref局部變數,並將ref關鍵字放在調用方法前面(ref在等式左右兩邊都有)。

ref int value = ref ElementAt(ref data, 2);

你還可以在不使用ref關鍵字的情況下調用此方法,這時候返回的是值。(譯註:不是引用)

int value = ElementAt(ref data, 2);

在這種情況下,程式輸出如下:

但是,你需要在兩邊都指定ref,或者兩邊都沒有ref;你不能在一邊指定ref而在另一邊不指定ref。

此外,以下代碼也可以運行:

ElementAt(ref data, 2) = 5;

你將獲得與第一個示例相同的輸出。位置2的元素從0(預設值)更改為5。這是因為ref返回可以作為一個LValue出現。(譯註:LValue被稱為左值,簡單地說就是出現在等號左邊的值,詳見"左值和右值表達式")

解決重載

因為返回類型不是認定一個重載的方法簽名的一部分,所以下麵的方法定義可能不起作用。(譯註:不能同時出現在一個類中,他們被視為同樣的簽名,同樣簽名的方法在一個類中只能出現一次)

public int M(int[] value);
public ref int M(int[] value);

但是下麵的定義將會起作用,因為參數在第一個方法中有ref,在第二個方法中沒有ref,是傳值。(譯註:參數是方法簽名的一部分)

public ref int M(ref int[] value);
public ref int M(int[] value);

因此,為了重載一個ref返回結果方法,你需要滿足如下參數規則:

  • 使用不同數量的參數,或者
  • 使用不同類型的參數,或者
  • 使用不同的傳值方式的參數(值或者引用)。

限制

ref return有一些限制:

  1. 一個方法ref返回的值必須是一個ref local(譯註:ref局部變數);這個ref局部變數的來源可以是這個方法的一個實際的ref/out參數,或者是聲明方法所在類型中的一個欄位。
  2. 不能引用返回空類型。
  3. 不能直接引用返回null。但是你可以返回一個ref局部變數,它的值是null。
  4. 不能從非同步方法ref返回,因為在非同步方法返回時,返回值可能是不確定的。
  5. 不允許ref返回常量和枚舉。

結論

ref返回結果是C#語言的擴展,它可以通過減少從方法返回中複製值的可能性來提高代碼的性能。這對於那些低級別的編程組件很有用,比如互操作性、跨平臺或資源受限的場景(移動、物聯網等)。要使用這個特性,你需要使用C# language 7.0,升級你的Visual Studio版本到2017年或更高版本。

 

系列文章:


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、下載(原文鏈接:http://www.studyshare.cn/software/details/1183/0 ) 1、官網下載:下載地址 2、百度網盤下載:下載地址 提取碼:0g5a java開發工具下載地址及安裝教程大全,點這裡。 更多深度技術文章,在這裡。 二、安裝及啟動 1、前提條件: ...
  • 本文將介紹在Java程式中如何將Excel工作簿轉為PDF文檔的,包括: 將整個工作簿轉為PDF 將指定工作表轉為PDF 使用工具:Free Spire.XLS for Java (免費版) Jar文件下載及導入: 方法1:通過官網下載。下載後,解壓文件,將lib文件夾下的Spire.Xls.jar ...
  • 場景 JPA入門簡介與搭建HelloWorld(附代碼下載): https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/103473937 JPA中實現單向多對一的關聯關係: https://blog.csdn.net/BADAO_LIUM ...
  • 一、Task類簡介: Task類是在.NET Framework 4.0中提供的新功能,主要用於非同步操作的控制。它比Thread和ThreadPool提供了更為強大的功能,並且更方便使用。 Task和Task<TResult>類:前者接收的是Action委托類型;後者接收的是Func<TResult ...
  • 問題 在開發一款應用的過程中,我們開發者很難考慮到所有問題,往往會忘記處理一些可能發生的異常。隨之而來的結果就是用戶使用過程中接連不斷的崩潰。所以,我們有必要處理所有未被我們處理的異常。 思路 我們需要做的是,在錯誤發生時保存用戶數據,然後將錯誤直接展示在用戶界面上。 解決 首先,我們打開項目中的 ...
  • 比如我們需要ASP.NET Core 中需要通過PDF來進行某些簡單的報表開發,隨著這並不難,但還是會手忙腳亂的去搜索一些資料,那麼恭喜您,這篇帖子會幫助到您,我們就不會再去浪費一些寶貴的時間。 在本文中我們將要使用DinkToPDF來處理我們在.NET Core Web 程式中進行構建PDF文檔! ...
  • EulerOS其實出來有一段時間了,一直在關註,單是僅僅也只是停留在觀望的階段,目前還沒有接入的打算;正好看到園子里的兄弟分享了華為雲免費試用的活動後,難捺激動的心情,我馬上去申請試用了一臺伺服器。 ...
  • 如果要在程式中使用DbContext,則需要先在Nuget中安裝Microsoft.EntityFrameworkCore.SqlServer ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...