tnblog
首页
视频
资源
登录

快醒醒,C# 9 中又来了一堆关键词 init,record,with

1426人阅读 2024/9/20 16:36 总访问:3658352 评论:0 收藏:0 手机
分类: .net后台框架

.netcore

快醒醒,C# 9 中又来了一堆关键词 init,record,with

新增关键词

init


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

  1. public class Person
  2. {
  3. public string Name { get; init; }
  4. }


乍一看有点懵逼,没关系,先用 ILSpy 看一下,如下图:


上面这张图就已经很清晰的解释了,原来 init 就是自动生成了一个对 私有只读字段 的封装,对于 readonly 相信大家已经轻车熟路了,它的初始化只有两种方式:声明时和构造函数中,但从 C# 9 开始就多了一个属性赋值方式,也就是说现在有三种赋值方式了,还原代码如下:

  1. public class Person
  2. {
  3. private readonly string name;
  4. public string Name
  5. {
  6. get => name;
  7. init
  8. {
  9. name = value;
  10. }
  11. }
  12. }


这种方式要是换作以前肯定是报错的,如下图:


有一点要注意的是编译器还做了一个特殊限制,准你在 类初始化器 中使用,不准你单独拿出来赋值,如下图所示:


所以总的来说, init 的作用就是多了一种让你初始化 只读字段 的方式,仅此而已罢了。

record


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

  1. public record Person
  2. {
  3. public string Name { get; set; }
  4. public int Age { get; set; }
  5. }


看起来挺 ???? 的,现在除了 class,struct , enum, delegate,又来了一个 record,俺们的 C# 是越来越强大啦。
还是老规矩,用ILspy看看底层生成了个啥,如下代码所示:

  1. public class Person : IEquatable<Person>
  2. {
  3. protected virtual Type EqualityContract => typeof(Person);
  4. public string Name
  5. {
  6. get;
  7. set;
  8. }
  9. public int Age
  10. {
  11. get;
  12. set;
  13. }
  14. public virtual Person <>Clone()
  15. {
  16. return new Person(this);
  17. }
  18. public override int GetHashCode()
  19. {
  20. return (EqualityComparer<Type>.Default.GetHashCode(EqualityContract) * -1521134295 + EqualityComparer<string>.Default.GetHashCode(Name)) * -1521134295 + EqualityComparer<int>.Default.GetHashCode(Age);
  21. }
  22. public override bool Equals(object? obj)
  23. {
  24. return Equals(obj as Person);
  25. }
  26. public virtual bool Equals(Person? P_0)
  27. {
  28. 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);
  29. }
  30. protected Person(Person P_0)
  31. {
  32. Name = P_0.Name;
  33. Age = P_0.Age;
  34. }
  35. public Person()
  36. {
  37. }
  38. bool IEquatable<Person>.Equals(Person other)
  39. {
  40. return Equals(other);
  41. }
  42. }


从 ILspy 生成出来的代码来看,可以发现两点信息:
————record 玩的也是 class,重写了 object 中的一些方法 GetHashCode, Equals 等等。
————按类中的字段逐一比较判断类的相等性。
说到根据字段判断类的相等性,不知道大家可有似曾相识的感觉? ,反正让我想起了匿名类型,因为它生成的 C# 代码和 record 如出一辙,不信的话,我演示给你看呗。

  1. var person = new { Name = "jack", Age = 20 };


接下来看一看是否真的是按照逐一字段比较,代码如下图:

  1. static void Main(string[] args)
  2. {
  3. var person = new Person() { Name = "jack", Age = 20 };
  4. var person2 = new Person() { Name = "jack", Age = 20 };
  5. var b = person.Equals(person2);
  6. }


看了这么多,我想你肯定有一些疑问:
1) 为啥要实现 IEquatable 接口
这是因为在当 Person 是 泛型 T 的时候避免走了默认的 public override bool Equals(object? obj),这是一个双装箱操作,性能太低效,深入研究可看我的博文:https://www.tnblog.net/hb/article/details/8463
2) 为啥有 equals 没有 ==
这个问题问得好,谁知道 C# 开发团队怎么想的,按照目前现状, 用 == 和 equals 比较两个对象,结果肯定是不一样的,我想你肯定能理解,毕竟一个是引用一个是按字段比较,这就比较坑爹了,如下图:

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


从方法体来看,这个方法用于做 浅copy 用的,但方法名前面有一对 <> ,说明是防你直接调用的,那问题来了,怎么调用呢? 这就涉及一个新的语法糖。

3. with


这个语法糖也挺????的,就是为了助你调用 record 的 <>clone 方法,不信的话,上代码呗。

  1. static void Main(string[] args)
  2. {
  3. var person = new Person() { Name = "jack", Age = 20 };
  4. var person2 = person with { };
  5. }


然后看一下 IL 反编译的代码


不过我也有一个疑问,为啥要防着我直接调用 Clone 方法呢? 新东西,也不知道应用场景,谁搞的清楚哈~~~ ??????


欢迎加群讨论技术,1群:677373950(满了,可以加,但通过不了),2群:656732739

评价
这一世以无限游戏为使命!
排名
2
文章
657
粉丝
44
评论
93
docker中Sware集群与service
尘叶心繁 : 想学呀!我教你呀
一个bug让程序员走上法庭 索赔金额达400亿日元
叼着奶瓶逛酒吧 : 所以说做程序员也要懂点法律知识
.net core 塑形资源
剑轩 : 收藏收藏
映射AutoMapper
剑轩 : 好是好,这个对效率影响大不大哇,效率高不高
ASP.NET Core 服务注册生命周期
剑轩 : http://www.tnblog.net/aojiancc2/article/details/167
ICP备案 :渝ICP备18016597号-1
网站信息:2018-2025TNBLOG.NET
技术交流:群号656732739
联系我们:contact@tnblog.net
公网安备:50010702506256
欢迎加群交流技术