快醒醒,C# 9 中又來了一堆關鍵詞 init,record,with

一:背景

1. 講故事

.NET5 終於在 2020-08-25 也就是大前天發佈了第八個預覽版,這麼多的預覽版搞得我都麻木了,接踵而來的就是更多的新特性加入到了 C# 9 中,既然還想呆在這條船上,得繼續硬着頭皮學習哈,這一篇跟你們聊聊新增的幾個關鍵詞。html

二:新增關鍵詞

1. init

出來一個新語法糖,首先要作的就是去揭它的老底,這樣能夠方便推測它的應用場景,爲了方便表述,我先上一個例子:jquery

public class Person
    {
        public string Name { get; init; }
    }

乍一看有點懵逼,不要緊,先用 ILSpy 看一下,以下圖:less

上面這張圖就已經很清晰的解釋了,原來 init 就是自動生成了一個對 私有隻讀字段 的封裝,對於 readonly 相信你們已經輕車熟路了,它的初始化只有兩種方式:聲明時和構造函數中,但從 C# 9 開始就多了一個屬性賦值方式,也就是說如今有三種賦值方式了,還原代碼以下:ide

public class Person
    {
        private readonly string name;

        public string Name
        {
            get => name;

            init
            {
                name = value;
            }
        }
    }

這種方式要是換做之前確定是報錯的,以下圖:函數

有一點要注意的是編譯器還作了一個特殊限制,準你在 類初始化器 中使用,不許你單獨拿出來賦值,以下圖所示:性能

因此總的來講, init 的做用就是多了一種讓你初始化 只讀字段 的方式,僅此而已罷了。學習

2. record

爲了方便演示,我先上一段代碼,以下所示:this

public record Person
    {
        public string Name { get; set; }

        public int Age { get; set; }
    }

看起來挺 🐂👃 的,如今除了 class,struct , enum, delegate,又來了一個 record,俺們的 C# 是愈來愈強大啦。code

仍是老規矩,用ILspy看看底層生成了個啥,以下代碼所示:htm

public class Person : IEquatable<Person>
{
	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<Type>.Default.GetHashCode(EqualityContract) * -1521134295 + EqualityComparer<string>.Default.GetHashCode(Name)) * -1521134295 + EqualityComparer<int>.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<string>.Default.Equals(Name, P_0!.Name) && EqualityComparer<int>.Default.Equals(Age, P_0!.Age);
	}

	protected Person(Person P_0)
	{
		Name = P_0.Name;
		Age = P_0.Age;
	}

	public Person()
	{
	}

	bool IEquatable<Person>.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),這是一個雙裝箱操做,性能過低效,深刻研究可看個人博文:http://www.javashuo.com/article/p-dslxakid-cp.html

2) 爲啥有 equals 沒有 ==

這個問題問得好,誰知道 C# 開發團隊怎麼想的,按照目前現狀, 用 == 和 equals 比較兩個對象,結果確定是不同的,我想你確定能理解,畢竟一個是引用一個是按字段比較,這就比較坑爹了,以下圖:

3) <>Clone() 方法有何做用

從方法體來看,這個方法用於作 淺copy 用的,但方法名前面有一對 <> ,說明是防你直接調用的,那問題來了,怎麼調用呢? 這就涉及一個新的語法糖。

3. with

這個語法糖也挺🐂👃的,就是爲了助你調用 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。 有一點要提醒的是,語法糖多了,必定要知道其實它是個啥,不要常年混在編譯器之上迷失了方向😄😄😄

如您有更多問題與我互動,掃描下方進來吧~

圖片名稱
相關文章
相關標籤/搜索