mirror of
https://github.com/m-lamonaca/dev-notes.git
synced 2025-04-06 10:56:41 +00:00
Improve record notes on immutability
This commit is contained in:
parent
4b5c170b11
commit
8a39b7085e
1 changed files with 35 additions and 21 deletions
|
@ -305,34 +305,54 @@ Since the names of the attributes of a tuple do not matter (a tuple is an instan
|
||||||
(int a, int b) = point; // extract data based on position into two variables
|
(int a, int b) = point; // extract data based on position into two variables
|
||||||
```
|
```
|
||||||
|
|
||||||
## Records (C# 9)
|
## Records
|
||||||
|
|
||||||
|
I'ts possible to create record types with immutable properties by using standard property syntax or positional parameters:
|
||||||
|
|
||||||
```cs
|
```cs
|
||||||
public record Record
|
public record Person
|
||||||
{
|
{
|
||||||
public Type Property1 { get; init; }
|
public string FirstName { get; init; } = default!;
|
||||||
public Type Property2 { get; init; }
|
public string LastName { get; init; } = default!;
|
||||||
|
};
|
||||||
|
// same as
|
||||||
|
public record Person(string FirstName, string LastName);
|
||||||
|
|
||||||
// Constructor, Deconstructor, Equals(), Equality Operators, GetHashCode(), ToString() generated
|
public record struct Point
|
||||||
|
{
|
||||||
|
public double X { get; init; }
|
||||||
|
public double Y { get; init; }
|
||||||
|
public double Z { get; init; }
|
||||||
}
|
}
|
||||||
// same as
|
// same as
|
||||||
public record Record(Type Property1, Type Property2, ...);
|
public readonly record struct Point(double X, double Y, double Z);
|
||||||
public record struct RecordStruct(Type Property1, Type Property2, ...);
|
|
||||||
|
|
||||||
Record recObj = new Record(value1, ...); // instantiation
|
|
||||||
//or
|
|
||||||
Record record = new Record { Property1 = value1, ...}; // instantiation
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### Custom Behaviour
|
it's also possible to create record types with mutable properties and fields:
|
||||||
|
|
||||||
```cs
|
```cs
|
||||||
public record Record(Type Property1, Type Property2, ...)
|
// mutable record
|
||||||
|
public record Person
|
||||||
{
|
{
|
||||||
// provide custom implementation for default maethod
|
public string FirstName { get; set; } = default!;
|
||||||
}
|
public string LastName { get; set; } = default!;
|
||||||
|
};
|
||||||
|
|
||||||
|
// mutable record struct
|
||||||
|
public readonly record struct Point(double X, double Y, double Z);
|
||||||
```
|
```
|
||||||
|
|
||||||
|
While records can be mutable, they're primarily intended for supporting immutable data models. The record type offers the following features:
|
||||||
|
|
||||||
|
- Concise syntax for creating a reference type with immutable properties
|
||||||
|
- Built-in behavior useful for a data-centric reference type:
|
||||||
|
- Value equality
|
||||||
|
- Concise syntax for nondestructive mutation
|
||||||
|
- Built-in formatting for display
|
||||||
|
- Support for inheritance hierarchies
|
||||||
|
|
||||||
|
Note: A _positional record_ and a _positional readonly record struct_ declare init-only properties. A _positional record struct_ declares read-write properties.
|
||||||
|
|
||||||
### `with`-expressions
|
### `with`-expressions
|
||||||
|
|
||||||
When working with immutable data, a common pattern is to create new values from existing ones to represent a new state.
|
When working with immutable data, a common pattern is to create new values from existing ones to represent a new state.
|
||||||
|
@ -352,12 +372,6 @@ protected Record(Record original) { /* copy all the fields */ } // generated
|
||||||
|
|
||||||
**NOTE**: it's possible to define a custom copy constructor tha will be picked up by the `with` expression.
|
**NOTE**: it's possible to define a custom copy constructor tha will be picked up by the `with` expression.
|
||||||
|
|
||||||
### Value-based Equality
|
|
||||||
|
|
||||||
All objects inherit a virtual `Equals(object)` method from the object class. This is used as the basis for the `Object.Equals(object, object)` static method when both parameters are non-null.
|
|
||||||
Structs override this to have "value-based equality", comparing each field of the struct by calling `Equals` on them recursively. Records do the same.
|
|
||||||
This means that in accordance with their "value-ness" two record objects can be equal to one another without being the same object.
|
|
||||||
|
|
||||||
### `with`-expressions & Inheritance
|
### `with`-expressions & Inheritance
|
||||||
|
|
||||||
Records have a hidden virtual method that is entrusted with "cloning" the whole object.
|
Records have a hidden virtual method that is entrusted with "cloning" the whole object.
|
||||||
|
|
Loading…
Add table
Reference in a new issue