2021-01-31 11:05:37 +01:00
# Rust
## Compiling and running
```ps1
rustc file.rs # compilation
file.exe # running
```
## Basics
```rs
use < module > ; // bring a type into scope
fn main() { //program entry point
// code here
}
```
### Result types
2021-09-20 19:35:32 +02:00
The `Result` types are **enumerations** , often referred to as *enums* . An enumeration is a type that can have a *fixed set of values* , and those values are called the enum's **variants** .
2021-01-31 11:05:37 +01:00
2021-08-13 19:03:14 +02:00
For `Result` , the variants are `Ok` or `Err` .
The `Ok` variant indicates the operation was *successful* , and inside `Ok` is the successfully generated value.
2021-01-31 11:05:37 +01:00
The `Err` variant means the operation failed, and `Err` contains information about how or why the operation failed.
The purpose of the `Result` types is to encode error-handling information.
Rust has a number of types named `Result` in its standard library: a generic `Result` as well as specific versions for submodules.
### Screen Output
```rs
// macro (funcs have no "!")
println!("Value: {}", value); // {} is a placeholder for a value or variable
2021-10-08 22:29:46 +02:00
println!("Values: {1}, {0}", value1, value2); // use index to print values
println!("Num: {:< total_digits > .< decimal_digits > }", 10);
println!("Num: {:0< total > .< decimal > }", 10); // prefix num with zeroes
println!("{:b}", 0b11011); // print as bits
2021-01-31 11:05:37 +01:00
print!();
```
### User Input
```rs
use io;
let mut string = String::new(); // create empty string var
let mut var: i32 = io:stdin().readline().trim().parse().expect("input must be a number");
io::stdin() // read line from stdin
.readline(& mut string) // put in into a var (since var i mutable it has to be specified with & mut)
.expect("Error Message"); // in case of errors return an error message (io::Result) -- NEEDED
```
2021-08-13 19:03:14 +02:00
The `::` syntax in the `::new` line indicates that `new` is an **associated function** of the `String` type.
An associated function is implemented on a type rather than on a particular instance of the type. Some languages call this a `static method` .
2021-01-31 11:05:37 +01:00
The `&` indicates that this argument is a reference, which gives a way to let multiple parts of the code access one piece of data without needing to copy that data into memory multiple times.
## Variables & Mutability
By default variables are *immutable* .
```rs
let var1 = value; // immutable var init
let var2: Type = value; // explicit type annotation
let mut var3 = value; // mutable var init
let mut var4: Type = value; // explicit type annotation
const CONSTANT_NAME: type = value; // constant must have the type annotation
```
### Shadowing
2021-08-13 19:03:14 +02:00
It's possible declare a new variable with the *same name* as a previous variable, and the new variable *shadows* the previous variable.
2021-01-31 11:05:37 +01:00
By using let, it's possible to perform a few transformations on a value but have the variable be immutable after those transformations have been completed.
2021-09-20 19:35:32 +02:00
The other difference between *mut* and *shadowing* is that because we're effectively creating a new variable when we use the let keyword again,
2021-08-13 19:03:14 +02:00
we can change the type of the value but reuse the same name.
2021-01-31 11:05:37 +01:00
```rs
let x: u32 = 10;
let x: i32 = 11; // shadowing
```
## Data Types
### Integer Types
| Length | Signed | Unsigned |
|--------------|---------|----------|
| 8-bit | `i8` | `u8` |
| 16-bit | `i16` | `u16` |
| 32-bit | `i32` | `u32` |
| 64-bit | `i64` | `u64` |
| 128-bit | `i128` | `u128` |
| architecture | `isize` | `usize` |
### Floating-Point Types
2021-08-13 19:03:14 +02:00
Rust also has two primitive types for floating-point numbers, which are numbers with decimal points.
2021-09-20 19:35:32 +02:00
Rust's floating-point types are `f32` and `f64` , which are 32 bits and 64 bits in size, respectively.
The default type is `f64` because on modern CPUs it's roughly the same speed as `f32` but is capable of more precision.
2021-01-31 11:05:37 +01:00
2021-10-08 22:48:35 +02:00
### Numeric OPerators
| Operator | Operation |
|----------|---------------------|
| `>` | Greater than |
| `>=` | Greater or Equal to |
| `<` | Less than |
| `<=` | Less or Equal to |
| `==` | Equals |
| `!=` | Not Equals |
### Comparison Operators
2021-01-31 11:05:37 +01:00
| Operator | Operation |
|----------|----------------|
| `+` | Addition |
| `-` | Subtraction |
| `*` | Multiplication |
| `/` | Division |
| `%` | Modulo |
### Boolean Types
2021-08-13 19:03:14 +02:00
Boolean types in Rust have two possible values: `true` and `false` .
2021-01-31 11:05:37 +01:00
Booleans are one byte in size. The Boolean type in Rust is specified using `bool` .
2021-10-08 22:48:35 +02:00
### Bitwise Operators
| Operator | Operation |
|----------|-------------------|
| `&` | AND |
| `|` | OR |
| `&&` | SHORT-CIRCUIT AND |
| `||` | SHORT-CIRCUIT OR |
| `^` | XOR |
| `!` | NOT |
| `<<` | LEFT SHIFT |
| `>>` | RIGHT SHIFT |
2021-01-31 11:05:37 +01:00
### Character Types
2021-09-20 19:35:32 +02:00
Rust's `char` type is the language's most primitive alphabetic type.
2021-01-31 11:05:37 +01:00
2021-09-20 19:35:32 +02:00
Rust's `char` type is four bytes in size and represents a Unicode Scalar Value: range from `U+0000` to `U+D7FF` and `U+E000` to `U+10FFFF` inclusive.
2021-01-31 11:05:37 +01:00
```rs
let c: char = 'C'; // SINGLE QUOTES
2021-10-08 22:53:37 +02:00
let c: char = '\u{261D}'; // Unicode Code Point U+261D
2021-01-31 11:05:37 +01:00
```
### String Types
```rs
let s = String::new(); // create empty string
let s = String::from("string literal"); // construct string from literal
s.push_str(""); // appending string literals
```
### Tuple Types
2021-08-13 19:03:14 +02:00
A tuple is a general way of grouping together a number of values with a variety of types into one compound type.
2021-01-31 11:05:37 +01:00
Tuples have a *fixed length* : once declared, they cannot grow or shrink in size.
```rs
let tup: (i32, f64, u8) = (500, 6.4, 1);
2021-10-09 11:05:59 +02:00
let tup = (500, 6.4, 1);
2021-01-31 11:05:37 +01:00
2021-09-20 19:35:32 +02:00
let (x, y, z) = tup; // tuple deconstruction (unpacking)
2021-01-31 11:05:37 +01:00
2021-10-09 11:05:59 +02:00
tup.0 = value; // member access & update (mut be mutable)
2021-01-31 11:05:37 +01:00
```
### Array Types
2021-08-13 19:03:14 +02:00
Every element of an array must have the *same type* . Arrays in Rust have a fixed length, like tuples.
2021-09-20 19:35:32 +02:00
An array isn't as flexible as the `vector` type, though. A vector is a similar collection type provided by the standard library that *is allowed to grow or shrink in size* .
2021-01-31 11:05:37 +01:00
```rs
let array = [0, 1, 2, 3, 4];
let array: [Type; length] = [...];
2021-10-09 11:05:59 +02:00
let array: [value; length]; // repeat expression (same as python's [value] * length)
2021-01-31 11:05:37 +01:00
2021-10-09 11:05:59 +02:00
let index: usize = < index_value > ; // indexes and ranges must be of type usize
array[index] = value; // member access and update (must be mutable)
let matrix = [
[0 ,1, 2],
[3, 4, 5]
]
let matrix: [[Type, length]; length]; = [[...], [...], ...]
matrix[row][column];
2021-01-31 11:05:37 +01:00
```
### Slice Types
2021-09-20 19:35:32 +02:00
Slices allows to reference a contiguous sequence of elements in a collection rather than the whole collection. **Slices don't take ownership** .
2021-01-31 11:05:37 +01:00
```rs
let s = String::from("string literal");
let slice = &s[start..end];
let a = [0, 1, 2, 3, 4, 5];
let slice = &a[start..end];
```
## Functions
Rust code uses *snake_case* as the conventional style for function and variable names.
2021-08-13 19:03:14 +02:00
Function definitions in Rust start with `fn` and have a set of parentheses after the function name.
2021-01-31 11:05:37 +01:00
The curly brackets tell the compiler where the function body begins and ends.
2021-09-20 19:35:32 +02:00
Rust doesn't care where the functions are defined, only that they're defined somewhere.
2021-01-31 11:05:37 +01:00
```rs
fn func(param: Type) { // parameters MUST have the Type annotation
// code here
}
fn func() -> Type { // -> specifies the return type
// code here
}
fn func() {
value // returns value
}
// same as
fn func() {
return value;
}
fn func() {
return (value1, value2, ...); // return multiple values with tuples
}
```
## Control Flow
### if - else if - else
```rs
if condition {
// code here
} else if condition {
// code here
} else {
// code here
}
```
### let if
```rs
let var = if condition { value } else { value }; // returned types must be the same
```
### loop
```rs
loop { // loop's forever if not explicitly stopped
// code here
if condition {
break returned_value
}
}
```
### while
```rs
while condition { // runs while condition is true
// code here
}
```
### for
```rs
for item in sequence.iter() {
// code here
}
for i in (start..end) { // (start..stop) is like python's range(start, stop)
// code here
}
```
## Ownership
2021-09-20 19:35:32 +02:00
Ownership is Rust's most unique feature, and it enables Rust to make memory safety guarantees without needing a garbage collector.
2021-01-31 11:05:37 +01:00
2021-09-20 19:35:32 +02:00
All programs have to manage the way they use a computer's memory while running.
2021-08-13 19:03:14 +02:00
Some languages have garbage collection that constantly looks for no longer used memory as the program runs; in other languages, the programmer must explicitly allocate and free the memory.
Rust uses a third approach: memory is managed through a system of ownership with a set of rules that the compiler checks at compile time.
2021-09-20 19:35:32 +02:00
None of the ownership features slow down your program while it's running.
2021-01-31 11:05:37 +01:00
### Stack & Heap
Both the stack and the heap are parts of memory that are available to your code to use at runtime, but they are structured in different ways.
2021-08-13 19:03:14 +02:00
The *stack* stores values in the order it gets them and removes the values in the opposite order. This is referred to as *last in, first out* .
2021-01-31 11:05:37 +01:00
Adding data is called *pushing* onto the stack, and removing data is called *popping* off the stack.
All data stored on the stack must have a known, fixed size. Data with an unknown size at compile time or a size that might change must be stored on the heap instead.
The heap is less organized: when you put data on the heap, you request a certain amount of space. The memory allocator finds an empty spot in the heap that is big enough, marks it as being in use, and returns a **pointer** , which is the address of that location. This process is called *allocating on the heap* and is sometimes abbreviated as just *allocating* .
2021-08-13 19:03:14 +02:00
Pushing to the stack is faster than allocating on the heap because the allocator never has to search for a place to store new data; that location is always at the top of the stack.
2021-01-31 11:05:37 +01:00
Comparatively, allocating space on the heap requires more work, because the allocator must first find a big enough space to hold the data and then perform bookkeeping to prepare for the next allocation.
Accessing data in the heap is slower than accessing data on the stack because you have to follow a pointer to get there. Contemporary processors are faster if they jump around less in memory.
2021-09-20 19:35:32 +02:00
Keeping track of what parts of code are using what data on the heap, minimizing the amount of duplicate data on the heap, and cleaning up unused data on the heap so you don't run out of space are all problems that ownership addresses.
2021-01-31 11:05:37 +01:00
### Ownership Rules
2021-09-20 19:35:32 +02:00
- Each *value* in Rust has a variable that's called its *owner* .
2021-01-31 11:05:37 +01:00
- There can only be one owner at a time.
- When the owner goes out of scope, the value will be dropped.
### Ways Variables and Data Interact
2021-08-13 19:03:14 +02:00
A "shallow copy" of a variable allocated on the heap (C#'s reference value) the original variable goes out of scope and only the "copy" remains.
2021-01-31 11:05:37 +01:00
A "deep copy" (`var.clone()` ) makes a copy of the data in the new variable without make the original fall out of scope.
2021-09-20 19:35:32 +02:00
When a variable goes out of scope, Rust calls a special function for us. This function is called `drop` , and it's where the code to return the memory is located.
2021-01-31 11:05:37 +01:00
2021-08-13 19:03:14 +02:00
Rust has a special annotation called the `Copy` trait that we can place on types that are stored on the stack.
2021-01-31 11:05:37 +01:00
If a type has the `Copy` trait, an older variable is still usable after assignment.
2021-09-20 19:35:32 +02:00
Rust won't let us annotate a type with the `Copy` trait if the type, or any of its parts, has implemented the `Drop` trait.
If the type needs something special to happen when the value goes out of scope and we add the `Copy` annotation to that type, we'll get a compile-time error.
2021-01-31 11:05:37 +01:00
```rs
let s = String::new()
let t = s; // shallow copy, s is now out of scope
let u = t.clone(); // deep copy, t is still valid
```
### Ownership & Functions
The semantics for passing a value to a function are similar to those for assigning a value to a variable. Passing a variable to a function will move or copy, just as assignment does.
```rs
fn main() {
let s = String::from("hello"); // s comes into scope
takes_ownership(s); // s's value moves into the function and so is no longer valid here
let x = 5; // x comes into scope
2021-09-20 19:35:32 +02:00
makes_copy(x); // x would move into the function, but i32 is Copy, so it's okay to still use x afterward
2021-01-31 11:05:37 +01:00
} // Here, x goes out of scope, then s. But because s's value was moved, nothing special happens.
fn takes_ownership(some_string: String) { // some_string comes into scope
println!("{}", some_string);
} // Here, some_string goes out of scope and `drop` is called. The backing memory is freed.
fn makes_copy(some_integer: i32) { // some_integer comes into scope
println!("{}", some_integer);
} // Here, some_integer goes out of scope. Nothing special happens.
```
### Return Values & Scope
Returning values can also transfer ownership.
```rs
fn main() {
let s1 = gives_ownership(); // gives_ownership moves its return value into s1
let s2 = String::from("hello"); // s2 comes into scope
let s3 = takes_and_gives_back(s2); // s2 is moved into takes_and_gives_back, which also moves its return value into s3
} // Here, s3 goes out of scope and is dropped. s2 goes out of scope but was moved, so nothing happens. s1 goes out of scope and is dropped.
fn gives_ownership() -> String { // gives_ownership will move its return value into the function that calls it
let some_string = String::from("hello"); // some_string comes into scope
some_string // some_string is returned and moves out to the calling function
}
// takes_and_gives_back will take a String and return one
fn takes_and_gives_back(a_string: String) -> String { // a_string comes into scope
a_string // a_string is returned and moves out to the calling function
}
```
### References & Borrowing
*Mutable references* have one big restriction: you can have *only one* mutable reference to a particular piece of data in a particular scope.
2021-08-13 19:03:14 +02:00
The benefit of having this restriction is that Rust can prevent **data races** at compile time.
2021-01-31 11:05:37 +01:00
A data race is similar to a race condition and happens when these three behaviors occur:
2021-08-13 19:03:14 +02:00
- Two or more pointers access the same data at the same time.
- At least one of the pointers is being used to write to the data.
2021-09-20 19:35:32 +02:00
- There's no mechanism being used to synchronize access to the data.
2021-01-31 11:05:37 +01:00
```rs
fn borrow(var: & Type) { // & Type indicates that the var is a reference
// here var cannot be modified
} // when var goes out of scope it doesn't get dropped because the scope didn't own it
borrow(&variable); // & variable creates a reference to variable but does not take ownership
fn borrow2(var: & mut Type) {
// here var can be modified
}
```
## Structs
A **struct** , or structure, is a custom data type that allows to name and package together multiple related values that make up a meaningful group.
2021-08-13 19:03:14 +02:00
To define a struct enter the keyword struct and name the entire struct.
2021-09-20 19:35:32 +02:00
A struct's name should describe the significance of the pieces of data being grouped together.
2021-01-31 11:05:37 +01:00
Then, inside curly brackets, define the names and types of the pieces of data, which we call *fields* .
```rs
struct Struct {
field: Type,
...
}
{ x: Type, y: Type } // anonymous struct
```
To use a struct after defining it, create an instance of that struct by specifying concrete values for each of the fields.
```rs
let mut var = Struct {
field: value,
...
};
fn build_struct(param: Type, ...) -> Struct {
2021-09-20 19:35:32 +02:00
// the constructed struct is returned since it's the last expression
2021-01-31 11:05:37 +01:00
Struct {
field: param,
...
}
}
```
### Field Init Shorthand
```rs
fn build_struct(field: Type, ...) -> Struct {
2021-09-20 19:35:32 +02:00
// the constructed struct is returned since it's the last expression
2021-01-31 11:05:37 +01:00
Struct {
field, // shortened form since func param is named as the struct's field
...
}
}
var.field = value; // member access
```
2021-09-20 19:35:32 +02:00
**Note**: the entire instance must be mutable; Rust doesn't allow to mark only certain fields as mutable.
2021-01-31 11:05:37 +01:00
### Struct Update Syntax
```rs
let struct1 = Struct {
field1: value,
field2: value,
...
}
// same as
let struct2 = Struct {
field1: other_value,
..struct1 // all remaining fields have the same values of struct1
}
```
### Using Tuple Structs without Named Fields to Create Different Types
To define a **tuple struct** , start with the `struct` keyword and the struct name followed by the types in the tuple.
```rs
struct Point(i32, i32, i32);
let origin = Point(0, 0, 0);
```
2021-09-20 19:35:32 +02:00
### Struct Printing
2021-01-31 11:05:37 +01:00
```rs
#[derive(Debug)] // inherit the debug traits
struct StructName
{
field: value,
...
}
2021-09-20 19:35:32 +02:00
let s: Struct = { /* valorization */};
2021-01-31 11:05:37 +01:00
printl!("{:?}", s) // debug output: { field: value, ... }
```
### Method Syntax
```rs
struct Struct
{
field: value,
...
}
impl Struct
{
fn method(& self, arg: Type) -> Type { }
}
2021-09-20 19:35:32 +02:00
let s: Struct = { /* valorization */};
s.method(arg); // use struct method
2021-01-31 11:05:37 +01:00
```
## Enums
```rs
// enum definition
enum Enum
{
Variant1(Type, ...), // each variant can have different types (even structs and enums) and amounts of associated data
Variant2,
...
}
2021-09-20 19:35:32 +02:00
// value assignment
2021-01-31 11:05:37 +01:00
let e: Enum = Enum::Variant2;
let e: Enum = Enum::Variant1(arg, ...); // variant w/ data
// methods on enum
impl Enum
{
fn method(& self, arg: Type) -> Type {}
}
```
### [Option enum](https://doc.rust-lang.org/std/option/enum.Option.htmls)
2021-09-20 19:35:32 +02:00
The `Option` type is used in many places because it encodes the very common scenario in which a value could be something or it could be nothing. Expressing this concept in terms of the type system means the compiler can check whether you've handled all the cases you should be handling; this functionality can prevent bugs that are extremely common in other programming languages.
2021-01-31 11:05:37 +01:00
2021-09-20 19:35:32 +02:00
he `Option<T>` enum is so useful that it's even included in the prelude; you don't need to bring it into scope explicitly.
2021-01-31 11:05:37 +01:00
In addition, so are its variants: you can use `Some` and `None` directly without the `Option::` prefix.
```rs
// std implementation
enum Option< T > {
Some(T),
None
}
```
2021-09-20 19:35:32 +02:00
*NOTE*: When `None` is used the type of `Option<T>` must be specified, because the compiler can't infer the type that the `Some` variant will hold by looking only at a `None` value.
2021-01-31 11:05:37 +01:00
### Match Expression + Comparing
2021-08-13 19:03:14 +02:00
A **match expression** is made up of *arms* .
2021-09-20 19:35:32 +02:00
An arm consists of a *pattern* and the code that should be run if the value given to the beginning of the match expression fits that arm's pattern.
Rust takes the value given to match and looks through each arm's pattern in turn.
2021-01-31 11:05:37 +01:00
2021-09-20 19:35:32 +02:00
**NOTE**: `match` arms must be exhaustive for compilation.
2021-01-31 11:05:37 +01:00
```rs
enum Enum {
Variant1,
Variant2,
Variant3(value),
...
}
// match expression on enum
fn match_variant(e: Enum) {
match e {
Enum::Variant1 => < expr > ,
Enum::Variant2 => { /* code block */ },
Enum::Variant3(value) => state,
_ => () // () is unit value
}
}
```
## Collections
### Vector
2021-02-24 22:02:12 +01:00
Vectors allow to store more than one value in a single data structure that puts all the values next to each other in memory. Vectors can only store values of the *same type* .
Like any other struct, a vector is freed when it goes out of scope. When the vector gets dropped, all of its contents are also dropped.
```rs
let v: Vec< Type > = Vec< Type > ::new(); // empty vec init
let mut v: vec![item1, item2, ...]; // vec init (type inferred)
v.push(item); // add elements to vector
// element access
v.get(index); // get method (returns Option< & T>)
2021-09-20 19:35:32 +02:00
&v[index]; // index syntax (returns reference, panic on index out of bounds)
2021-02-24 22:02:12 +01:00
// iterate over mutable references to each element in a mutable vector in order to make changes to all the elements
for i in mut & v {
*i = value; // dereference and modify value
}
```
A vector can hold different types if those type are variants of the same enum. It's also possible to use trait objects.
```rs
enum Enum {
Int(i32),
Float(f64),
Text(String)
}
let v = vec![
Enum::Int(2),
Enum::Float(3.14),
Enum::Text("TEST")
];
```