C#9新關鍵詞詳解
導讀 | .NET5 終於在 2020-08-25 也就是前天釋出了第八個預覽版,這麼多的預覽版搞得我都麻木了,接踵而來的就是更多的新特性加入到了 C# 9 中,既然還想呆在這條船上,得繼續硬著頭皮學習哈,這一篇跟大家聊聊新增的幾個關鍵詞。 |
.NET5 終於在 2020-08-25 也就是前天釋出了第八個預覽版,這麼多的預覽版搞得我都麻木了,接踵而來的就是更多的新特性加入到了 C# 9 中,既然還想呆在這條船上,得繼續硬著頭皮學習哈,這一篇跟大家聊聊新增的幾個關鍵詞。
出來一個新語法糖,首先要做的就是去揭它的老底,這樣可以方便推測它的應用場景,為了方便表述,我先上一個例子:
public class Person { public string Name { get; init; } }
乍一看有點懵逼,沒關係,先用 ILSpy 看一下,如下圖:
上面這張圖就已經很清晰的解釋了,原來 init 就是自動生成了一個對 私有隻讀欄位 的封裝,對於 readonly 相信大家已經輕車熟路了,它的初始化只有兩種方式:宣告時和建構函式中,但從 C# 9 開始就多了一個屬性賦值方式,也就是說現在有三種賦值方式了,還原始碼如下:
public class Person { private readonly string name; public string Name { get => name; init { name = value; } } }
這種方式要是換作以前肯定是報錯的,如下圖:
有一點要注意的是編譯器還做了一個特殊限制,準你在 類初始化器 中使用,不准你單獨拿出來賦值,如下圖所示:
所以總的來說, init 的作用就是多了一種讓你初始化 只讀欄位 的方式,僅此而已罷了。
為了方便演示,我先上一段程式碼,如下所示:
public record Person { public string Name { get; set; } public int Age { get; set; } }
看起來挺 ?? 的,現在除了 class,struct , enum, delegate,又來了一個 record,俺們的 C# 是越來越強大啦。
還是老規矩,用ILspy看看底層生成了個啥,如下程式碼所示:
public class Person : IEquatable { protected virtual Type EqualityContract => typeof(Person); public string Name { get; set; } public int Age { get; set; } public virtual Person <>Clone() { return new Person(this); } public override int GetHashCode() { return (EqualityComparer.Default.GetHashCode(EqualityContract) * -1521134295 + EqualityComparer.Default.GetHashCode(Name)) * -1521134295 + EqualityComparer.Default.GetHashCode(Age); } public override bool Equals(object? obj) { return Equals(obj as Person); } public virtual bool Equals(Person? P_0) { return P_0 != null && (object)EqualityContract == P_0!.EqualityContract && EqualityComparer.Default.Equals(Name, P_0!.Name) && EqualityComparer.Default.Equals(Age, P_0!.Age); } protected Person(Person P_0) { Name = P_0.Name; Age = P_0.Age; } public Person() { } bool IEquatable .Equals(Person other) { return Equals(other); }}
從 ILspy 生成出來的程式碼來看,可以發現兩點資訊:
record 玩的也是 class,重寫了 object 中的一些方法 GetHashCode, Equals 等等。
按類中的欄位逐一比較判斷類的相等性。
說到根據欄位判斷類的相等性,不知道大家可有似曾相識的感覺? ,反正讓我想起了匿名型別,因為它生成的 C# 程式碼和 record 如出一轍,不信的話,我演示給你看唄。
var person = new { Name = "jack", Age = 20 };
接下來看一看是否真的是按照逐一欄位比較,程式碼如下圖:
static void Main(string[] args) { var person = new Person() { Name = "jack", Age = 20 }; var person2 = new Person() { Name = "jack", Age = 20 }; var b = person.Equals(person2); }
看了這麼多,我想你肯定有一些疑問:
1) 為啥要實現 IEquatable 介面
這是因為在當 Person 是 泛型 T 的時候避免走了預設的 public override bool Equals(object? obj),這是一個雙裝箱操作,效能太低效,深入研究可看我的博文:https://www.cnblogs.com/huangxincheng/p/12996361.html 。
2) 為啥有 equals 沒有 ==
這個問題問得好,誰知道 C# 開發團隊怎麼想的,按照目前現狀, 用 == 和 equals 比較兩個物件,結果肯定是不一樣的,我想你肯定能理解,畢竟一個是引用一個是按欄位比較,這就比較坑爹了,如下圖:
3) <>Clone() 方法有何作用
從方法體來看,這個方法用於做 淺copy 用的,但方法名前面有一對 <> ,說明是防你直接呼叫的,那問題來了,怎麼呼叫呢?這就涉及一個新的語法糖。
這個語法糖也挺??的,就是為了助你呼叫 record 的 <>clone 方法,不信的話,上程式碼唄。
static void Main(string[] args) { var person = new Person() { Name = "jack", Age = 20 }; var person2 = person with { }; }
然後看一下 IL 反編譯的程式碼
不過我也有一個疑問,為啥要防著我直接呼叫 Clone 方法呢?新東西,也不知道應用場景,誰搞的清楚哈~~~ ???
總的來說C#是越來越新穎了,也一直在踐行 jquery 的口號:write less,do more。有一點要提醒的是,語法糖多了,一定要知道其實它是個啥,不要常年混在編譯器之上迷失了方向.
原文來自:
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69955379/viewspace-2719542/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- HanLP 關鍵詞提取演算法分析詳解HanLP演算法
- Java jvm級別native關鍵詞、JNI詳解JavaJVM
- 關鍵詞加粗和插入關鍵詞
- Java關鍵詞synchronized解讀Javasynchronized
- robot 關鍵詞
- 關鍵詞提取
- 匹配關鍵詞和敏感詞
- 網站關鍵詞堆砌後,處理關鍵詞堆砌方法網站
- java—— finall 關鍵詞Java
- 什麼是長尾關鍵詞?如何找到長尾關鍵詞?
- 高亮:單關鍵詞、多關鍵詞、多組多關鍵詞,從簡單到複雜實現滿足多方面需求的頁面關鍵詞高亮
- 什麼是關鍵詞策略?網站關鍵詞佈局重要嗎?網站
- 如何優化多個關鍵詞?分享多關鍵詞優化心得優化
- Go 切片詳解(理解是關鍵)Go
- C++ typeid關鍵字詳解C++
- 作用域鏈this關鍵詞
- Eclipse註釋關鍵詞Eclipse
- 關鍵詞感知檢索
- transient關鍵詞的概述
- c# yield關鍵字原理詳解C#
- C語言中static關鍵字詳解C語言
- Pig 實現關鍵詞匹配
- 帝國CMS列表頁模板新聞關鍵詞帶連結呼叫
- AI繪畫怎麼寫關鍵詞?AI繪畫高畫質桌布關鍵詞分享AI
- 網遊運營關鍵資料詳解
- 萬字乾貨|Synchronized關鍵字詳解synchronized
- Hive常用命令,快鍵和關鍵詞Hive
- HanLP-實詞分詞器詳解HanLP分詞
- [譯] 深入淺出 JavaScript 關鍵詞 -- thisJavaScript
- 如何用Python提取中文關鍵詞?Python
- 從JavaScript 的關鍵詞談起JavaScript
- 正規表示式關鍵詞解析
- SnowNLP——獲取關鍵詞(keywords(1))
- 使用Python呼叫API介面獲取京東關鍵詞詳情資料PythonAPI
- 讀懂這幾個關鍵詞,你就能瞭解 Docker 啦Docker
- 掌握seo關鍵點,輕鬆穩定網站關鍵詞排名網站
- 高效的關鍵詞替換和敏感詞過濾工具
- Java 10 var關鍵字詳解和示例教程Java