一晃距C# 9
發布已經4年了,對于record
關鍵字想必大家都不陌生了,不過呢發現還是有很多同學不屑于使用這個語法糖
,確實,本質上 record 就是 class 的封裝,能用 record 書寫的類,那100%都是可以自己手擼出來的,但是呢有沒有考慮 別人可能一分鐘寫好的代碼你可能會需要數分鐘才能完成.因此為了能有更多時間 摸魚 ,強烈推薦不屑一顧的同學也能用起來!
下面我簡略聊一聊 record 的好處和最佳場景:
- 簡化語法
我們只需要一行代碼就可以定義完成,這個是最直觀節省編碼的方式,我們不需要編寫一堆枯燥的get;set; 也不需要編寫構造函數等樣板代碼:
public record Person(string FirstName, string LastName);
那么有同學會有疑問,如果Person有很多的屬性咋整,不就意味著主構造函數會很冗長,其實呢,這個和封裝傳參的方式是一樣的,我們可以把同質的屬性封裝成其他的record或者class,比如:
public record ExtraInfomation(string Address,string Email,int Age);
public record Person(string FirstName, string LastName, ExtraInfomation ExtraInfo);
- 自動生成一些對我們有用的成員函數.
- 構造函數:根據定義的屬性自動生成構造函數。
- 屬性:自動生成只讀屬性。
- Deconstruct 方法:用于解構記錄對象,對于習慣寫
TS
的小伙伴相當友好。 - Equals 和 GetHashCode 方法:基于屬性值的相等性比較。
- ToString 方法:提供友好的字符串表示,對于調試輸出特別友好。
- 基于值的相等性語法.
我們很多時候有這種需求就是比較一個類的所有屬性來判斷邏輯.如果使用 record 的話 我們只需要==
或者Equals
就能判斷,
- 非破壞性復制值
對于一個 class 的淺表復制,我們可能需要實現ICloneable
,亦或者 new 一個對象逐個屬性賦值,當然還有其他的方法,但是呢肯定是沒有 record 來的這么簡單直接.我們僅需要一個with
關鍵字就干完了
public record Person(string FirstName, string LastName, int Age);
var person1 = new Person("vip", "wan", 18);
var person2 = person1 with { Age = 30 };
Console.WriteLine(person1);
Console.WriteLine(person2);
在單元測試中的場景:
public readonly record struct RegexMatch(string Raw, string Act);
[Theory]
[ClassData(typeof(AHrefTestData))]
public void Test_Regex_A_Href(RegexMatch rm)
{
var regex = new Regex(@"<a[^>]+href=(['""])(?<href>.*?)\1[^>]*>", RegexOptions.IgnoreCase);
var match = regex.Match(rm.Raw);
Assert.True(match.Success);
Assert.Equal(rm.Act, match.Groups["href"].Value);
}
public class AHrefTestData : TheoryData<RegexMatch>
{
public AHrefTestData()
{
var rm = new RegexMatch("<a , "http://www.baidu.com");
Add(rm);
Add(rm with { Raw = "<a href=\"http://www.baidu.com\"></a>" });
Add(rm with { Raw = "<a });
Add(rm with { Raw = "<a Href=\"http://www.baidu.com\">" });
}
}
- 解構的支持
record 類型自動生成 Deconstruct 方法,允許你輕松地解構 record 對象,對于全棧的同學書寫就是手到擒來!
var person = new Person("vip", "wan", 18);
var (firstName, lastName, age) = person;
Console.WriteLine(firstName);
Console.WriteLine(lastName);
Console.WriteLine(age);
- 結合模式匹配
record 類型與模式匹配功能很好地集成在一起,使得在模式匹配中使用 record 對象更加方便。
public record Person(string UserName, int Age);
public string GetPersonInfo(Person person) => person switch
{
{ Age: < 18 } => "Minor",
{ Age: >= 18 } => "Adult",
_ => "Unknown"
};
- 填充既有類
嗯當前 C# 語言是真的突飛猛進,年底就要發布C# 13
了,小伙伴們都直呼學不動了!,當然也有同學肯定也嘗鮮了主構造函數
了吧, 如果想要對主構造函數進一步了解可以 點擊鏈接 對于注入的服務又能少擼不少的代碼!
那么既然 class 都有了主構造函數
,是不是意味著 record 就失去意義了呢?!,嗯?!你忘了上面的那些糖的甜度了嗎?
因此我們如果需要對既有的 class 支持到 record 的特性我們只需要在class前加上 record 即可.
public record class User {
public string UserName{ get; set;}
public int Age { get; set;}
}
var user1 = new User { UserName = "vipwan" , Age = 18};
var user2 = user1 with { };
var user3 = user1 with { Age = 30 };
user1 == user2
user3.ToString()
C# 10
提供的record struct
,readonly record struct
支持:
默認情況下編譯器將record
等價于record class
,record class由于是基于class的封裝因此完整的繼承了class的多態性等特征,兩者復制比較 編譯器內部實現代碼是不同的,struct 的性能會稍好(值類型和引用類型的主要區別),因此MS在 C# 10中帶來了(readonly) record struct
的支持;
對于readonly record struct
和record struct
的區別:
- record struct:默認不可變,但可以包含可變字段和屬性,適用于需要一定可變性的值類型數據結構。
- readonly record struct:由于其完全不可變性,編譯器可以進行更多的優化,例如避免不必要的復制,從而提高性能。
對于代碼的區別請看:
public abstract record PersonBase(string FirstName, string LastName);
public record Person(string FirstName, string LastName, string Address) : PersonBase(FirstName, LastName)
{
public int? Age { get; set; }
}
public record struct PersonStruct(string FirstName, string LastName)
{
public int? Age { get; set; }
}
public readonly record struct PersonReadonlyStruct(string FirstName, string LastName)
{
public int? Age { get; init; }
}
對于如何選擇總結一句:
struct 極致性能, class 包容性強,對于大多數情況下 (readonly) record struct 夠用;對于包容免除后顧之憂,優先選擇 record class !
總結
使用 record 類型的主要好處包括簡潔的語法、自動生成的成員、基于值的相等性、非破壞性復制、解構支持、繼承支持和與模式匹配的良好集成。這些特性使得 record 類型非常適合用于不可變數據對象(DTO,VO等),提高了代碼的可讀性、可維護性和開發效率。
?轉自https://www.cnblogs.com/vipwan/p/18325508
該文章在 2024/10/16 10:22:34 編輯過