Skip to content

Commit

Permalink
Reword introduction of pattern matching
Browse files Browse the repository at this point in the history
* Add new introduction to irrefutable patterns.
* Move introduction of `match` to day 2.
  • Loading branch information
randomPoison committed Feb 28, 2024
1 parent d75dd5d commit e098799
Show file tree
Hide file tree
Showing 4 changed files with 41 additions and 29 deletions.
1 change: 1 addition & 0 deletions book.toml
Original file line number Diff line number Diff line change
Expand Up @@ -233,6 +233,7 @@ use-boolean-and = true
'unsafe/extern-functions.html' = '../unsafe-rust/unsafe-functions.html'
'unsafe/unsafe-traits.html' = '../unsafe-rust/unsafe-traits.html'
'exercises/day-3/safe-ffi-wrapper.html' = '../../unsafe-rust/exercise.html'
'tuples-and-arrays/match.html' = '../pattern-matching/match.html'

[output.exerciser]
output-directory = "comprehensive-rust-exercises"
4 changes: 2 additions & 2 deletions src/SUMMARY.md
Original file line number Diff line number Diff line change
Expand Up @@ -49,8 +49,7 @@
- [Tuples and Arrays](tuples-and-arrays.md)
- [Tuples and Arrays](tuples-and-arrays/tuples-and-arrays.md)
- [Array Iteration](tuples-and-arrays/iteration.md)
- [Pattern Matching](tuples-and-arrays/match.md)
- [Destructuring](tuples-and-arrays/destructuring.md)
- [Patterns and Destructuring](tuples-and-arrays/destructuring.md)
- [Exercise: Nested Arrays](tuples-and-arrays/exercise.md)
- [Solution](tuples-and-arrays/solution.md)
- [References](references.md)
Expand All @@ -73,6 +72,7 @@

- [Welcome](welcome-day-2.md)
- [Pattern Matching](pattern-matching.md)
- [Matching Values](pattern-matching/match.md)
- [Destructuring](pattern-matching/destructuring.md)
- [Let Control Flow](pattern-matching/let-control-flow.md)
- [Exercise: Expression Evaluation](pattern-matching/exercise.md)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
minutes: 10
---

# Pattern Matching
# Matching Values

The `match` keyword lets you match a value against one or more _patterns_. The
comparisons are done from top to bottom and the first match wins.
Expand All @@ -24,7 +24,7 @@ fn main() {
```

The `_` pattern is a wildcard pattern which matches any value. The expressions
_must_ be irrefutable, meaning that it covers every possibility, so `_` is often
_must_ be exhaustive, meaning that it covers every possibility, so `_` is often
used as the final catch-all case.

Match can be used as an expression. Just like `if`, each match arm must have the
Expand Down
61 changes: 36 additions & 25 deletions src/tuples-and-arrays/destructuring.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,44 +2,55 @@
minutes: 5
---

# Destructuring
# Patterns and Destructuring

Destructuring is a way of extracting data from a data structure by writing a
pattern that is matched up to the data structure, binding variables to
subcomponents of the data structure.

You can destructure tuples and arrays by matching on their elements:

## Tuples
When working with tuples and other structured values it's common to want to
extract the inner values into local variables. This can be done manually by
directly accessing the inner values:

```rust,editable
fn main() {
describe_point((1, 0));
fn print_tuple(tuple: (i32, i32)) {
let left = tuple.0;
let right = tuple.1;
println!("left: {left}, right: {right}");
}
```

However, Rust also supports using pattern matching to destructure a larger value
into its constituent parts:

fn describe_point(point: (i32, i32)) {
match point {
(0, _) => println!("on Y axis"),
(_, 0) => println!("on X axis"),
(x, _) if x < 0 => println!("left of Y axis"),
(_, y) if y < 0 => println!("below X axis"),
_ => println!("first quadrant"),
}
```rust,editable
fn print_tuple(tuple: (i32, i32)) {
let (left, right) = tuple;
println!("left: {left}, right: {right}");
}
```

## Arrays
This works with any kind of structured value:

```rust,editable
{{#include ../../third_party/rust-by-example/destructuring-arrays.rs}}
struct Foo {
a: i32,
b: bool,
}
fn print_foo(foo: Foo) {
let Foo { a, b } = foo;
println!("a: {a}, b: {b}");
}
```

<details>

- Create a new array pattern using `_` to represent an element.
- Add more values to the array.
- Point out that how `..` will expand to account for different number of
elements.
- Show matching against the tail with patterns `[.., b]` and `[a@..,b]`
- The patterns used here are "irrefutable", meaning that the compiler can
statically verify that the value on the right of `=` has the same structure as
the pattern.
- A variable name is an irrefutable pattern that always matches any value, hence
why we can also use `let` to declare a single variable.
- Rust also supports using patterns in conditionals, allowing for equality
comparison and destructuring to happen at the same time. This form of pattern
matching will be discussed in more detail later.
- Edit the examples above to show the compiler error when the pattern doesn't
match the value being matched on.

</details>

0 comments on commit e098799

Please sign in to comment.