C# 9 中新加入的关键词 init,record,with

一:背景

1. 讲故事

.NET5 终于在 2020-08-25 也就是大前天发布了第八个预览版,这么多的预览版搞得我都麻木了,接踵而来的就是更多的新特性加入到了 C# 9 中,既然还想呆在这条船上,得继续硬着头皮学习哈,这一篇跟大家聊聊新增的几个关键词。

二:新增关键词

1. init

出来一个新语法糖,首先要做的就是去揭它的老底,这样可以方便推测它的应用场景,为了方便表述,我先上一个例子:

 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 的作用就是多了一种让你初始化 只读字段 的方式,仅此而已罢了。

2. record

为了方便演示,我先上一段代码,如下所示:

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

  public int Age { get; set; }
 }

看起来挺 🐂👃 的,现在除了 class,struct , enum, delegate,又来了一个 record,俺们的 C# 是越来越强大啦。

还是老规矩,用ILspy看看底层生成了个啥,如下代码所示:

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),这是一个双装箱操作,性能太低效,深入研究可看我的博文:https://www.jb51.net/article/194342.htm

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。 有一点要提醒的是,语法糖多了,一定要知道其实它是个啥,不要常年混在编译器之上迷失了方向

以上就是C# 9 中新加入的关键词 init,record,with的详细内容,更多关于c# 9 新关键词的资料请关注我们其它相关文章!

(0)

相关推荐

  • 解答“60k”大佬的19道C#面试题(下)

    在上篇中,我解析了前 10 道题目,本篇我将尝试解析后面剩下的所有题目. 姐妹篇:解析"60k"大佬的19道C#面试题(上) 这些题目确实不怎么经常使用,因此在后文中,我会提一组我的私房经典"6k面试题",供大家轻松一刻. 先略看题目: 11 简述 LINQ 的 lazy computation 机制 12 利用 SelectMany 实现两个数组中元素做笛卡尔集,然后一一相加 13 请为三元函数实现柯里化 14 请简述 ref struct 的作用 15 请简述

  • 浅谈C# 9.0 新特性之只读属性和记录

    大家好,这是 C# 9.0 新特性系列的第 4 篇文章. 熟悉函数式编程的童鞋一定对"只读"这个词不陌生.为了保证代码块自身的"纯洁",函数式编程是不能随便"弄脏"外来事物(参数.变量等)的,所以"只读"对函数式编程非常重要. 为了丰富 C# 对函数式编程支持,较新的 C# 版本引入了一些很有用的新特性.比如 C# 8 中就对 struct 类型的方法增加了 readonly 修饰符支持,被 readonly 修饰的方法是不能

  • 浅析C# 9.0 新特性之 Lambda 弃元参数

    大家好,这是 C# 9.0 新特性短系列的第 5 篇文章. 弃元(Discards) 是在 C# 7.0 的时候开始支持的,它是一种人为丢弃不使用的临时虚拟变量.语法上它是用来赋值的,但它却不被分配存储空间,即没有值,所以不能从中读取值.弃元用 _(下划线) 表示,下划线是一个关键字,只能赋值,不能读取,例如: 在 C# 7.0 中,弃元的使用场景主要有下面四种: 元组和对象的解构 使用 is 和 switch 的模式匹配 对具有 out 参数的方法的调用 作用域内独立使用场景 针对这几个场景,

  • C#中的9个“黑魔法”

    我们知道C#是非常先进的语言,因为是它很有远见的"语法糖".这些"语法糖"有时过于好用,导致有人觉得它是C#编译器写死的东西,没有道理可讲的--有点像"黑魔法". 那么我们可以看看C#这些高级语言功能,是编译器写死的东西("黑魔法"),还是可以扩展(骚操作)的"鸭子类型". 我先列一个目录,大家可以对着这个目录试着下判断,说说是"黑魔法"(编译器写死),还是"鸭子类型"

  • 浅谈C#9.0新特性之参数非空检查简化

    参数非空检查是缩写类库很常见的操作,在一个方法中要求参数不能为空,否则抛出相应的异常.比如: public static string HashPassword(string password) { if(password is null) { throw new ArgumentNullException(nameof(password)); } ... } 当异常发生时,调用者很容易知道是什么问题.如果不加这个检查,可能就会由系统抛出未将对象引用为实例之类的错误,这不利于调用者诊断错误. 由

  • 使用Visual Studio2019创建C#项目(窗体应用程序、控制台应用程序、Web应用程序)

    一.VS的开发环境 首先你得安装了vs2019,然后确认下下面三个组件是否存在,如果没有要下载一下.vs2019的安装可参考visual studio2019的安装以及使用. 二.创建C#窗体应用程序 打开vs 可以直接在搜索框输入关键字进行搜索,选择 Windows 窗体应用(.NET Framework).如图. 也可以限定项目类型,所用的语言进行查找. 接下去狂点确定即可.当然可以选择改个项目存放存放的目录和项目名称. 创建好后,就进入到了这么一个界面. 在窗体右边(也可能是在左边.每个人

  • C# 获取某个时间的0点0分和23点59分59秒

    C# 获取某个时间的0点0分和23点59分59秒,具体代码如下所示: C#获取当月第一天和最后一天 当月第一天0时0分0秒: DateTime.Now.AddDays(1 - DateTime.Now.Day).Date 当月最后一天23时59分59秒: DateTime.Now.AddDays(1 - DateTime.Now.Day).Date.AddMonths(1).AddSeconds(-1) C#获取上个月第一天和最后一天 上个月第一天0时0分0秒: DateTime.Now.Add

  • c# 9.0新特性nint和Pattern matching的使用方法

    一:背景 1. 讲故事 上一篇跟大家聊到了Target-typed new 和 Lambda discard parameters,看博客园和公号里的阅读量都达到了新高,甚是欣慰,不管大家对新特性是多头还是空头,起码还是对它抱有一种极为关注的态度,所以我的这个系列还得跟,那就继续开撸吧,今天继续带来两个新特性,更多新特性列表,请大家关注:新特性预览 二:新特性研究 1. Native ints 从字面上看貌似是什么原生类型ints,有点莫名其妙,还是看一看Issues上举得例子吧: Summar

  • C# 9.0 新特性之模式匹配简化的实现

    记得在 MS Build 2020 大会上,C# 语言开发项目经理 Mads Torgersen 宣称 C# 9.0 将会随着 .NET 5 在今年 11 月份正式发布.目前 .NET 5 已经到了 Preview 5 阶段了,C# 9.0 也已经初具规模.忍不住激动的心情,暂停更新<C#.NET 拾遗补漏>系列几天,先要和大家分享一下我了解到的 C# 9.0 的新特性.由于新特性比较多,所以会分成几篇来讲.这是第一篇,专讲模式匹配这个特性的简化. 模式匹配(Pattern Matching)

  • 解答“60k”大佬的19道C#面试题(上)

    先略看题目: 1  请简述async函数的编译方式 2  请简述Task状态机的实现和工作机制 3  请简述await的作用和原理,并说明和GetResult()有什么区别 4  Task和Thread有区别吗?如果有请简述区别 5  简述yield的作用 6  利用IEnumerable<T>实现斐波那契数列生成 7  简述stackless coroutine和stackful coroutine的区别,并指出C#的coroutine是哪一种 8  请简述SelectMany的作用 9 

随机推荐