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