mirror of
https://github.com/m-lamonaca/dev-notes.git
synced 2025-04-05 18:36:41 +00:00
feat(rust): add concurrency notes
This commit is contained in:
parent
9cf06eea5e
commit
bcd9fcd591
1 changed files with 52 additions and 2 deletions
|
@ -1382,8 +1382,8 @@ impl<T> Drop for CustomSmartPointer<T> {
|
|||
}
|
||||
|
||||
fn main() {
|
||||
let var1 = CCustomSmartPointer(value); // dropped when var1 goes out of scope
|
||||
let var2 = CCustomSmartPointer(value);
|
||||
let var1 = CustomSmartPointer(value); // dropped when var1 goes out of scope
|
||||
let var2 = CustomSmartPointer(value);
|
||||
drop(var2); // dropped early by using std::mem::drop
|
||||
}
|
||||
```
|
||||
|
@ -1450,6 +1450,56 @@ Rust's memory safety guarantees make it difficult, but not impossible, to accide
|
|||
Rust allows memory leaks by using `Rc<T>` and `RefCell<T>`: it's possible to create references where items refer to each other in a cycle.
|
||||
This creates memory leaks because the reference count of each item in the cycle will never reach 0, and the values will never be dropped.
|
||||
|
||||
## Concurrency
|
||||
|
||||
### Creating Threads
|
||||
|
||||
The `thread::spawn` function creates a new tread that will execute the passed closure. Any data captured by the closure is _moved_ to the capturing thread.
|
||||
The thread's **handle** can be used to wait for completion and retieve the computation result or errors if any.
|
||||
|
||||
```rs
|
||||
// if no data is captured the move keyword can be removed
|
||||
let handle = std::thread::spawn(move || { /* ... */ });
|
||||
|
||||
handle.join().unwrap();
|
||||
```
|
||||
|
||||
> **Note**: a thread's execution can be termiated early if it's not joined and the main process terminates before the thread had completed it's work.
|
||||
> **Note**: if a thread panics the handle will return the panic message so that it can be handled.
|
||||
|
||||
### Channnels
|
||||
|
||||
To accomplish message-sending concurrently Rust's standard library provides an implementation of _channels_.
|
||||
A **channel** is a general programming concept by which data is sent from one thread to another.
|
||||
|
||||
```rs
|
||||
let (sender, receiver) = std::sync::mpsc::channel(); // the sender can be cloned to create multiple transmitters
|
||||
|
||||
let sender_1 = sender.clone();
|
||||
std::thread::spawn(move || {
|
||||
// send takes ownership of the message (moved to receiver scope)
|
||||
sender_1.send("hello".to_owned()).unwrap();
|
||||
});
|
||||
|
||||
let sender_2 = sender.clone();
|
||||
std::thread::spawn(move || {
|
||||
sender_2.send("hello".to_owned()).unwrap();
|
||||
sender_2.send("world".to_owned()).unwrap();
|
||||
sender_2.send("from".to_owned()).unwrap();
|
||||
sender_2.send("thread".to_owned()).unwrap();
|
||||
});
|
||||
|
||||
let message = receiver.recv().unwrap(); // receive a single value
|
||||
// or
|
||||
for message in receiver { } // receive multiple values (iteration stops when channel closes)
|
||||
```
|
||||
|
||||
### `Send` & `Sync`
|
||||
|
||||
The `Send` marker trait indicates that ownership of values of the type implementing `Send` can be transferred between threads. Any type composed entirely of `Send` types is automatically marked as `Send`. Almost all primitive types are `Send`, aside from raw pointers.
|
||||
|
||||
The `Sync` marker trait indicates that it is safe for the type implementing `Sync` to be referenced from multiple threads. In other words, any type `T` is `Sync` if `&T` (an immutable reference to `T`) is `Send`, meaning the reference can be sent safely to another thread. Similar to `Send`, primitive types are `Sync`, and types composed entirely of types that are `Sync `are also `Sync`.
|
||||
|
||||
## Files
|
||||
|
||||
### Reading Files
|
||||
|
|
Loading…
Add table
Reference in a new issue