您好,登錄后才能下訂單哦!
C# 9.0中怎么設置只讀屬性,針對這個問題,這篇文章詳細介紹了相對應的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。
為了豐富 C# 對函數式編程支持,較新的 C# 版本引入了一些很有用的新特性。比如 C# 8 中就對 struct 類型的方法增加了 readonly 修飾符支持,被 readonly 修飾的方法是不能修改該方法所在類的屬性的。舉個例子:
public struct FooValue
{
private int A { get; set; }
public readonly int IncreaseA()
{
A = A + 1; // 報錯
return A;
}
}
而 C# 9 又進一步增加了對“只讀”的支持,此次增加了 init-only 屬性和 record 相關特性,下面一一介紹。
我們知道類的屬性有 set 和 get 兩種訪問器,現在 C# 9 增加一種屬性訪問器:init。init 是 set 訪問器的變體,它的作用是使屬性只能在對象初始化的時候對其賦值,之后該屬性就是只讀的,因此叫 init-only 屬性。使用方式如下:
public class Foo
{
public string PropA { get; init; }
public string PropB { get; init; }
}
賦值操作:
var foo = new Foo { PropA = "A", PropB = "B" };
foo.PropA = "AA"; // 報錯,PropA 此時是只讀的!
由于 init 是在初始化階段賦值,所以它可以在類內部修改 readonly 修飾的字段。比如:
public class Foo
{
private readonly string propA;
private readonly string propB;
public string PropA
{
get => propA;
init => propA = (value ?? throw new ArgumentNullException(nameof(propA)));
}
public string PropA
{
get => propB;
init => propB = (value ?? throw new ArgumentNullException(nameof(propB)));
}
}
如果你知道在構造函數中可以對只讀字段/屬性賦值就自然也理解這一點。
做過財務系統的人都知道交易記錄一旦入賬是不能修改的,如果錄入錯誤,就要新錄入一筆負的記錄把之前的紅沖掉,再錄入正確的記錄。應對類似這種只讀記錄的場景,C# 9 引入了 Record(記錄,下文均使用中文的“記錄”)的概念,它用來支持整個對象的只讀特性(即實例化后為只讀)。使用方式如下:
public data class Foo
{
public string PropA { get; init; }
public string PropB { get; init; }
}
這里用了一個 data 關鍵字,表示該類的對象只是純粹的記錄值,它不是可修改的狀態(在函數式編程中,所有的數據修改都是狀態在發生變化)。
上面的太麻煩了,可以這樣簡寫:
public data class Foo
{
string PropA;
string PropB;
}
默認屬性都是 public 的,如果實在要改為 private,可以在屬性定義前面加上 private 修飾符。
有時候為了初始化更方便,可以定義構造函數來給屬性賦值,初始化時只需要把屬性值按順序傳給構造函數即可,這個操作稱為定位構造(Positional Construction)。同樣,也可以使用解構函數(Deconstructor)來實現屬性的解構,即按照解構函數的參數順序從對象中提取屬性的值,被稱為定位解構(Positional Deconstructor)。實現了定位構造或定位解構的記錄稱為定位記錄(Positional Record)。下面是一個定位記錄的實現:
public data class Foo
{
string PropA;
string PropB;
public Foo(string propA, string propB)
=> (PropA, PropB) = (propA, propB);
public void Deconstruct(out string propA, out string propB)
=> (propA, propB) = (PropA, PropB);
}
這個寫法太麻煩了,可以直接簡寫為:
public data class Foo(string PropA, string PropB);
這樣簡短一句代碼,其內部默認實現了 init-only 自動屬性,且同時為所有屬性定義了構造函數和解構函數。
使用示例:
var foo = new Foo("AA", "BB"); // 構造定位
var (a, b) = foo; // 解構定位
可以想象,記錄的大部分使用場景,以上簡寫的寫法能滿足需求。若有特殊場景,就不能簡單,需要進行自定義修改其默認行為。
當處理不可變數據時,若要生成不同的狀態,一個常見的場景是在一條舊記錄基礎上拷貝一條新的記錄。比如我們要修改 Foo 對象的 PropA 屬性,我們就要拷貝該對象生成一個新的對象。這個操作在函數式編程中被稱為“非破壞性修改 (non-destructive mutation)”。為了支持記錄的這個操作,C# 9 引入了 with 表達式,它可以很方便在一條原有記錄基礎上創建一條新記錄。示例:
var other = foo with { PropA = "AA" };
with 表達式內部其實是通過一個默認的 protected 構造函數來實現的,大致如下:
protected Foo(Foo original)
{
// 拷貝 original 的所有字段
}
關于C# 9.0中怎么設置只讀屬性問題的解答就分享到這里了,希望以上內容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關注億速云行業資訊頻道了解更多相關知識。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。