C#8.0 可空引用型別

MASA技術團隊發表於2022-01-23

介紹

我們的專案程式碼執行時最頻繁的錯誤之一就是 System.NullReferenceException 異常,c#8.0增加的可為空引用型別就是用來幫助開發者降低甚至消除NULL異常。我們需要注意的是可空引用型別是語法級別的功能,也就是程式碼編寫的時候就會受到程式設計約束,這個與可為空值型別是不一樣的。專案支援c#8.0請參見C# 語言版本控制

目錄

在專案中啟用可空引用型別支援

<Project Sdk="Microsoft.NET.Sdk.Web">

  <PropertyGroup>
    <TargetFramework>net6.0</TargetFramework>
    <Nullable>enable</Nullable>
  </PropertyGroup>
    
</Project>

在專案檔案中增加<Nullable>enable</Nullable>後,專案程式碼中的引用型別將被解析拆分為不可空引用型別可空引用型別

將警告提升為異常

可空引用型別功能是以警告的形式出現,並不會干擾專案生成編譯,約束力較弱。如果想嚴格要求自身,那我們可將特定的警告變為異常來提升約束力。

<Project Sdk="Microsoft.NET.Sdk.Web">

  <PropertyGroup>
    <TargetFramework>net6.0</TargetFramework>
    <Nullable>enable</Nullable>   				           
    <WarningsAsErrors>
    	$(WarningsAsErrors);CS8600;CS8601;CS8602;CS8603;CS8604;CS8609;CS8610;CS8614;CS8616;CS8618;CS8619;CS8622;CS8625
    </WarningsAsErrors>
  </PropertyGroup>
    
</Project>

相關技術文件C# 編譯器選項 - 錯誤和警告 | Microsoft DocsNon-nullable references with C# 8 and .NET Core 3.0 · Cezary Piątek Blog (cezarypiatek.github.io),大家在編寫程式碼時遇到Microsoft.CodeAnalysis.CSharp分析器所給的警告程式碼,都可按照自己的要求將其變為異常來約束自己。

將變數標註為可空引用型別

我們平時使用的引用型別屬於不可空引用型別,在其後附加?便為可空引用型別

string name; //不可空字串
string? adress; //可空字串

泛型

public TKey GetKey<TKey>()
{
    //必須返回不可空型別
}

public TValue? GetValue<TValue>()
{
    //可返回可空型別
}

使用示例

image-20211114144233735.png

如上示例,由於Student擁有預設的空建構函式new Student(),此建構函式會使NameAdress屬性為null,所以分析器發出了CS8618的警告。

image-20211114144541808.png

我們將空建構函式寫上,此時警告智慧的轉移到建構函式上了。

image-20211114145041104.png

我們在建構函式中將可能為null的string型別屬性附上值,警告消除。而string?型別無需處理,因為它是允許為null的。

image-20211114145405887.png

image-20211114145525241.png

以上兩種方式也可以消除警告。

image-20211114150135945.png

GetStudentNames方法中,我們使用StudentEnglishName屬性時,分析器發出了CS8604警告,因為EnglishName屬性是可空引用型別,無法放入List<string>中,只能放入在List<string?>中。

image-20211114150426034.png

我們使用??判斷當EnglishName為null時,使用不可空引用型別屬性Name,此時CS8604警告消除。

進階

可空引用型別模式中,屬性是可以被拆分為兩種模式的,其一是屬性是否可被賦值null,其二是屬性的值是否可能為null。大家可能對這句話理解起來有點懵,請接著看下面的講解。

[AllowNull]

不可為null的引用型別屬性允許被賦值null

image-20211114152033299.png

上面程式碼中,Adress屬性即使被賦值null,也不會使其值為null,不會在程式碼中引發潛在的Null異常。所以此場景是合理且被允許的。

[DisallowNull]

可為null的引用型別屬性不允許賦值為null

image-20211114152750169.png

Adress屬性雖然預設值是null,但對其賦值null是不合理的。雖然不能賦值null,但獲取Adress屬性的值時仍可能為null,大家可在合適的場景使用[DisallowNull]

[NotNull]

可為null的引用型別屬性的值永遠不會是null,可放心使用

image-20211114153459266.png

image-20211114153655036.png

我們使用GetStudentAdress方法返回StudentAdress屬性,分析器並沒有發出警告,因為分析器通過[NotNull]特性也知道了Adress屬性的值永遠不會為null。

image-20211114154005789.png

我們嘗試將Adress屬性改為可能返回null值,分析器立馬發出了CS8603警告,很給力。

[NotNullIfNotNull]

這個特性作用於方法中,用於告訴其他程式設計師只要你不給我的方法傳null參,我就不會返回null給你,你看著辦。

[return: NotNullIfNotNull("student")]
public string? GetStudentAdress(Student? student)
{
    return student?.Adress;
}

image-20211114154858318.png

adressadress2有著不同的待遇。

缺陷

有些場景分析器無法分析出潛在的null異常

Struct

public struct Student
{
    public string FirstName;
    public string? MiddleName;
    public string LastName;
}

public static class Program
{
    public static void PrintStudent(Student student)
    {
        Console.WriteLine($"First name: {student.FirstName.ToUpper()}");
        Console.WriteLine($"Middle name: {student.MiddleName?.ToUpper()}");
        Console.WriteLine($"Last name: {student.LastName.ToUpper()}");
    }

    public static void Main() => PrintStudent(default(FirstName));
    public static void Main2() => PrintStudent(new Student());
}

default(FirstName)new Student()中的FirstNameLastName 執行時為 null,編輯器此時未出現任何警告。

public struct Foo<T>
{
    public T Bar { get; set; }
}

public static class Program
{
    public static void Main()
    {
        string s = default(Foo<string>).Bar;
        string s2 = new Foo<string>().Bar;
    }
}

屬性 Bar 在執行時為 null,而ss2是不可為null字串型別,編輯器此時未出現任何警告。

陣列

陣列也是可為 null 的引用型別中的已知缺陷

using System;

public static class Program
{
    public static void Main()
    {
        string[] values = new string[10];
        string s = values[0];
        Console.WriteLine(s.ToUpper());
    }
}

程式碼中的陣列宣告其元素為不可為null的string,而其元素在初始化時都為null,編輯器此時未出現任何警告。

總結

將引用型別拆分為可空引用型別和不可空引用型別可以為我們的專案程式碼帶來質的提升,團隊之間協作或者使用第三方的類庫都可以通過?標識來知道方法的某個引數傳null不會引發異常、屬性賦值null不會引發異常,反之我們使用某些屬性或者方法的返參也可以知道其是否可能為null,對於不可能為null的變數我們就無需再麻煩的檢測null值了,而在以前,我們可能需要對每個變數都需要做null判斷。感興趣的同學趕緊給自己的專案加入這個功能吧。

我們正在行動,新的框架、新的生態

我們的目標是自由的易用的可塑性強的功能豐富的健壯的

所以我們借鑑Building blocks的設計理念,正在做一個新的框架MASA Framework,它有哪些特點呢?

  • 原生支援Dapr,且允許將Dapr替換成傳統通訊方式
  • 架構不限,單體應用、SOA、微服務都支援
  • 支援.Net原生框架,降低學習負擔,除特定領域必須引入的概念,堅持不造新輪子
  • 豐富的生態支援,除了框架以外還有元件庫、許可權中心、配置中心、故障排查中心、報警中心等一系列產品
  • 核心程式碼庫的單元測試覆蓋率90%+
  • 開源、免費、社群驅動
  • 還有什麼?我們在等你,一起來討論

經過幾個月的生產專案實踐,已完成POC,目前正在把之前的積累重構到新的開源專案中

目前原始碼已開始同步到Github(文件站點在規劃中,會慢慢完善起來):

MASA.BuildingBlocks

MASA.Contrib

MASA.Utils

MASA.EShop

BlazorComponent

MASA.Blazor

QQ群:7424099

微信群:加技術運營微信(MasaStackTechOps),備註來意,邀請進群

masa_stack_tech_ops.png

​ ------ END ------

作者簡介

吳煒來:MASA技術團隊成員。

相關文章