前言

之前在 C# – 10.0 已经有介绍过 Record 了. 但之前还没怎么用到, 最近有用到了, 所以特别写多一篇.

Class vs Struct

参考: C#详解struct和class的区别

它们最大的区别在于 Class 是引用类型, Struct 是值类型. 引用类型 (heap) vs 值类型 (stack)

定义是差不多的, 可以有 property, method, constructor 等等

public struct DimensionStruct
{
public int Width { get; set; }
public int Height { get; set; }
} public class DimensionClass
{
public int Width { get; set; }
public int Height { get; set; }
}

在对比值和引用的时候就看出区别了

var dimensionClass = new DimensionClass { Width = 100, Height = 100 };
var dimensionClass2 = dimensionClass;
var d = Object.ReferenceEquals(dimensionClass, dimensionClass2); // true
var e = Object.Equals(dimensionClass, dimensionClass2); // true
var f = Object.Equals(dimensionClass, new DimensionClass { Width = 100, Height = 100 }); // false 虽然里面 value 一样

实例指向同一个地址, Object.Equals 对比的是它们的地址是否一致, 而不是值是否一致.

在看 Struct

var dimensionStruct = new DimensionStruct { Width = 100, Height = 100 };
var dimensionStruct2 = dimensionStruct;
var a = Object.ReferenceEquals(dimensionStruct, dimensionStruct2); // false
var b = Object.Equals(dimensionStruct, dimensionStruct2); // true
var c = Object.Equals(dimensionStruct, new DimensionStruct { Width = 100, Height = 100 }); // true 值一样就行了

地址肯定是不一样的了. Object.Equals 对比的是里面的值. 值一样就行了.

When to use Struct?

我个人是没有特别感觉什么情况非用 Struct 不可.

比较常见的是 Size (width, height), Coordinate (x, y) 这类的 object value 就会用 struct.

Record

参考:

Intro to Records in C# 9 - How To Use Records And When To Use Them

Record Structs

Record Structs

Record 是 9.0 出来的. 先看看它的特色.

Definition

首先是它的 Definition 可以很短

public record Dimension(int Width, int Height);

3 大特色

record 有 3 大特色

1. immutable

函数式提倡的东西. 对象创建后, 属性值就不可以修改了.

var dimension = new Dimension(Width: 100, Height: 200);
dimension.Width = 5; // Error: record 是 immutable

上面这样是直接 complie error 的.

2. Deconstruct

类似 JavaScript 的解构, 它利用 Tuple 解构的特性.

var dimension = new Dimension(Width: 100, Height: 200);
var (width, height) = dimension; // base on sequence // int width;
// (width, var height) = dimension; // 赋值给 existing variable // var (_, height) = dimension; // underscore for skip

对比 JS 还是输一点, 没有那么灵活.

3. assign and clone

immutable 要怎样修改 value 呢? 靠的是 clone 一个新的来赋值.

var dimension = new Dimension(Width: 100, Height: 200);
var newDimension = dimension with { Height = 100 };

比起 JS 还是输很多. 但至少是跨出去了.

Record vs Class

首先要知道 Record 背地里就是 Class 实现来的,只是 compiler 做了手脚而已,不信可以去 sharplab.io 自己看看。

我们先讲雷同的地方. 上面 3 大特色, Class 可以完成 immutable 和 Deconstruct

public class DimensionClass
{
public DimensionClass(int width, int height)
{
Width = width;
Height = height;
} public int Width { get; init; } // 在 setter 使用关键字 init 就可以实现 immutable 了
public int Height { get; init; } internal void Deconstruct(out int width, out int height)
{
width = Width;
height = Height;
} // 这样写也可以
// internal void Deconstruct(out int width, out int height) => (width, height) = (Width, Height);
}

使用

var dimensionClass = new DimensionClass(width: 100, height: 200);
// dimensionClass.Width = 5; // error
var (width, height) = dimensionClass;

关键就在 setter 的 init 和 Deconstruct 方法.

主要的区别是 Class 的 Definition 比起 Record 太繁琐了.

但是, 如果 Record 你想要添加 Constructor 的话, 它就会变得和 Class 一模一样繁琐了.

public record Dimension
{
public Dimension(int width, int height)
{
Width = width;
Height = height;
}
public int Width { get; init; }
public int Height { get; set; } // 定义 setter 的话, Record 就不是 immutable 了哦 // 因为不是用默认的 constructor 所以需要另外定义 Deconstruct
internal void Deconstruct(out int width, out int height) => (width, height) = (Width, Height);
}

setter 用 set 的话可以让 record 变成不是 immutable, 当有了 constructor 就需要另外定义 Deconstruct 方法了.

所以 Record 和 Class 的区别在 immutable 和 Deconstruct 其实差不多.

只是 Record 有一些 default 逻辑, 在简单的案例中会比 Class 更方便定义.

Record 比 Class 厉害的是第三特点 assign and clone. 关键字 with 只能用在 record. Class 无法做到.

但是 ! 在 C# 10.0 之后, 多了一个 mix. 叫 record class.

加上 record 关键字后, 这个 Class 就可以使用 with 关键字了. 至此 Class 和 Record 更像了.

Object.Equals(record)

Class 和 Record 都是引用类似. 但是 Object.Equals 对 Class, 是对比 reference, 而对 record 和 record class 是对比值.

When to use?

如果你有喜欢 immutable 和 with 这类特性, 那么就可以考虑用 record.

至于是 record 还是 record class. 够简单就用 record. 复杂一点的话就尽量用 record class.

Record class overload constructor error

遇到一个 error, 不清楚什么原因, 先记入在这里.

A copy constructor in a record must call a copy constructor of the base, or a parameterless object constructor if the record inherits from object. [TestRImage]csharp(CS8868)

引发的原因是 record class + overload constructor + parameter is class object + calling this

目前没空检查. 解决方法就是不要 call this.

 

C# – Record, Class, Struct的更多相关文章

  1. 简明python教程 --C++程序员的视角(八):标准库

    os模块 这个模块包含普遍的操作系统功能. 如果你希望你的程序能够与平台无关的话,这个模块是尤为重要的.一个例子就是使用os.sep可以取代操作系统特定的路径分割符. os.system() 执行li ...

  2. https那些事儿

    (一)SSL/TLS协议运行机制的概述 一.作用 不使用SSL/TLS的HTTP通信,就是不加密的通信.所有信息明文传播,带来了三大风险. (1) 窃听风险(eavesdropping):第三方可以获 ...

  3. Windows SEH学习 x86

    windows 提供的异常处理机制实际上只是一个简单的框架.我们通常所用的异常处理(比如 C++ 的 throw.try.catch)都是编译器在系统提供的异常处理机制上进行加工了的增强版本.这里先抛 ...

  4. 函数lock_rec_add_to_queue

    在原来的type_mode基础上,加上LOCK_REC /*********************************************************************// ...

  5. C 高级编程5 IO与文件权限

    三.基于文件的描术符号 .得到文件描术符号/释入文件描术符号 a.文件类型 目录文件 d 普通文件 f 字符设务文件 c 块设备文件 b 软连接文件 l 管道文件 p socket文件 s 字符设备文 ...

  6. 函数lock_rec_find_similar_on_page

    /*********************************************************************//** Looks for a suitable type ...

  7. LR(1)表生成算法演示程序

    /* * LR 转换表 * + Goto 记录表 * + 状态转换表 */ #include <stdio.h> #include <stdlib.h> #include &l ...

  8. 简单的LR核心项集和Goto表填充演示程序

    /* * 该程序用于计算语言的核心项集 * RexfieldVon * 2013年8月24日21:19:25 */ #include <stdio.h> #include <stdl ...

  9. B树——思路、及C语言代码的实现

    0.序 本人现读本科大二,这学期学习数据结构,老师为我们的期末作业布置一道任选题,而我一直以来都有听说B树是一棵挺神奇的树,所以我选择了它,当然更重要的原因是因为B树的难度最高,我喜欢做有挑战性的工作 ...

  10. 一个简单的RPC框架

    一个 系统模型 二.数据库代码实现 1. mkdir database cd database vim dbInit.c /* * * Database Init tool * */ #include ...

随机推荐

  1. 让你的vscode搭载ChatGPT获得来自 AI 的编程指导

    一直以来,VS Code 都是开发者心目中的生产力神器,它免费.开源且跨平台,被称为最好用的 IDE. 把 VS Code 和 ChatGPT 结合使用,用户将获得来自 AI 的编程指导,包括代码解释 ...

  2. PowerBuilder现代编程方法X11:PB程序完全跨平台方案

    PB可能要支持Windows.macOS.Linux.iOS.Android与鸿蒙操作系统和X86.ARM.RISC-V与国产龙芯CPU的原生应用了! PowerBuilder现代编程方法X11:PB ...

  3. oeasy教您玩转python - 008 - # ascii码表

    ​ ASCII 码表 回忆上次内容 通过 help()可以从 python 命令行模式进入到帮助模式 通过 q 退出 ord(c)和 chr(i) 这是俩函数 这俩是一对,相反相成的 ord 通过字符 ...

  4. 题解:B3646 数列前缀和 3

    分析 板子题,线段树维护矩阵区间积,除了难写没什么思维难度. 所以直接放代码吧. Code #include<bits/stdc++.h> #define int long long us ...

  5. TIER 0: Meow

    TIER 0: Meow Virtual Machine 虚拟机 (VM) 指通过软件模拟完整计算机系统 可以实现环境隔离 模型宿主机不具备的架构 虚拟机的工作原理:在物理机器的操作系统上,以一个应用 ...

  6. 机器学习:详解多任务学习(Multi-task learning)

    详解多任务学习 在迁移学习中,步骤是串行的,从任务\(A\)里学习只是然后迁移到任务\(B\).在多任务学习中,是同时开始学习的,试图让单个神经网络同时做几件事情,然后希望这里每个任务都能帮到其他所有 ...

  7. WPF控件库

    Welcome to CookPopularControl 介绍 CookPopularControl是支持.NetFramework4.6.1与.Net5.0的WPF控件库,其中参考了一些资料,目前 ...

  8. 7、SpringMVC之RESTful概述

    创建名为spring_mvc_rest的新module,过程参考5.2节和6.6节 7.1.简介 RESTful 也称为REST(英文:Representational State Transfer) ...

  9. OpenCV计算机视觉学习(16)——仿射变换学习笔记

    如果需要其他图像处理的文章及代码,请移步小编的GitHub地址 传送门:请点击我 如果点击有误:https://github.com/LeBron-Jian/ComputerVisionPractic ...

  10. linux工具grep的使用心得笔记

    grep作为linux管理中常用的三大工具之一(grep.awk.sed),其功能十分强大,因此难以对其进行全面的使用介绍,因此本文只作为个人学习的笔记之用. grep的用处: 在文本中匹配要查询的字 ...