cargo run --example enums
cargo run --example iterator
...
- Install
$ curl --proto '=https' --tlsv1.2 https://sh.rustup.rs -sSf | sh
- Update using
$ rustup update
- View installed version via
$ rustup show
- Check latest version via
$ rustup check
Often,
cargo check
is much faster thancargo build
, because it skips the step of producing an executable. If you’re continually checking your work while writing the code, usingcargo check
will speed up the process! As such, many Rustaceans runcargo check
periodically as they write their program to make sure it compiles. Then they runcargo build
when they’re ready to use the executable.
fn main() {
println!("Hello World!");
}
$ rustc hello.rs
$ ./hello
..
used for range like1..4
i.e. 1, 2, 3. But, if1..=4
i.e. 1, 2, 3, 4- There are different types of struct
- normal struct: with parameters
- unit struct: without parameters
“Ownership is Rust’s most unique feature, and it enables Rust to make memory safety guarantees without needing a garbage collector.”
Borrow Checker: You can move the data itself and give up ownership in the process, create a copy of the data and pass that along, or pass a reference to the data and retain ownership, letting the recipient borrow it for a while. The most appropriate approach depends entirely on the situation. Try this
- Stack (fixed size like char, bool, int; less costly; quick to access by calling var like easy to copy the var) | Heap (variable size like string, list, class; more costly; access var or object via pointer)
- By default, all the variables are defined as
immutable
equivalent toconst
in JS/TS. - In Rust, borrowing is analogous to referencing in C++ & dereferencing is same as that of C++.
- The value of mutable variable can be changed, but not the type.
- In Rust, every value has a single owner that determines its lifetime.
- The memory of the declared variables are dropped (or freed) when the program leaves a block in which the variable is declared.
- E.g. Normally, inside the
main
function, whenever a variable is defined, it is dropped after exiting themain
function.
- E.g. Normally, inside the
fn main() {
// Case-1
let x = 10;
let r = &x;
let k;
{
let y = Box::new(5); // Using Box pointer for storing into heap
let y = 5; // stored in stack
// let y <'a> = 5;
// k = &y; // y dropped here as it is not available for lifetime. Moreover the block is getting over after this
k = y; // this implies that the ownership of 5 is transferred to `k` from `y`
}
}
- Rust doesn't allow dangling pointer by design. This means that any variable, struct, enum, etc can't live more than the lifetime of the referenced type
struct Config {
}
// INCORRECT ❌
struct App {
config: &Config // `Config` used as reference
}
// CORRECT ✅
/// Here, it is used as lifetime ownership of the code.
struct App<'a> {
config: &'a Config
}
lifetimes
are a compile-time feature and don’t exist at runtime.- Rust memory safety is based on this rule: Given an object T, it is only possible to have one of the following:
- Having several immutable references (&T) to the object (also known as aliasing).
- Having one mutable reference (&mut T) to the object (also known as mutability).
- Apply
#[derive(Debug)]
for making the struct, enum printable - Apply
#[derive(Clone)]
for making the struct, enum copyable. Option
vsResult
Option | Result |
---|---|
Some or None | Ok or Err |
An optional value can have either Some value or no value/ None. | A result can represent either success/ Ok or failure/ Err |
The Option type is a way to use Rust’s type system to express the possibility of absence | Result expresses the possibility of error |
mainly used for var, function output. For struct, the parameters can have Option type. E.g. In full name, middle_name can be missing for cases, so define middle_name: Option<String> |
mainly used for operation, function. As normally a variable won't have Err unless there is some calculation involved with this var |
Don't want to print the exact issue as None doesn't have anything as param unlike Some(T) |
Want to print the exact issue as Err(E) contains the message inside |
E.g. "./tuts/error_handling/opt" | E.g. "./tuts/error_handling/res" |
- Various sizes of integers, signed and unsigned (i32, u8, etc.)
- Floating point types f32 and f64.
- Booleans (bool)
- Characters (char). Note these can represent unicode scalar values (i.e. beyond ASCII)
-
- formatting variables inside
println
function
- formatting variables inside
let name = "Abhijit";
let age = 28;
println!("My name is {name}, and age is {age}"); // ❌
println!("My name is {0}, and age is {1}", name, age); // ✔️
println!("My name is {}, and age is {}", name, age); // ✔️
-
- Multiple usage of variables without repetition
let alice = "Alice";
let bob = "Bob";
println!("{0}, this is {1}. {1}, this is {0}", alice, bob);
- Check behind-the-code for a code snippet - https://play.rust-lang.org/
- Tools >> Expand Macros
- Cause: there is a statement having no effect
- Solution: Assign the variable to
_
.
Before:
let result = match grade {
"A" => { println!("Excellent!"); },
"B" => { println!("Great!"); },
"C" => { println!("Good"); },
"D" => { println!("You passed"); },
"F" => { println!("Sorry, you failed"); },
_ => { println!("Unknown Grade"); }
};
result;
After:
let result = match grade {
"A" => { println!("Excellent!"); },
"B" => { println!("Great!"); },
"C" => { println!("Good"); },
"D" => { println!("You passed"); },
"F" => { println!("Sorry, you failed"); },
_ => { println!("Unknown Grade"); }
};
// result; // warning: path statement with no effect, Solution --> assign to `_`
let _ = result;
- Cause: It simply means that the variant is never used, "constructed", anywhere in your program. There is no
AppAction::Task
anywhere in the program. Rust expects that if you say an enum variant exists, you will use it for something somewhere. - Solution: by putting this before the enum, or individually before intentionally unused items, you can make the warning disappear:
Before:
enum UsState {
California,
Mexico,
Alaska,
}
enum Coin {
Penny,
Nickel,
Dime,
Quarter,
Custom(UsState),
}
After:
#[allow(dead_code)]
#[derive(Debug)] // this use is recommended, otherwise there is error.
enum UsState {
California,
Mexico,
Alaska,
}
#[allow(dead_code)]
enum Coin {
Penny,
Nickel,
Dime,
Quarter,
Custom(UsState),
}
- Cause: Copy designates types for which making a bitwise copy creates a valid instance without invalidating the original instance.
This isn't true for String, because String contains a pointer to the string data on the heap and assumes it has unique ownership of that data. When you drop a String, it deallocates the data on the heap. If you had made a bitwise copy of a String, then both instances would try to deallocate the same memory block, which is undefined behaviour.
- Solution: Just use
format
like this:
Before:
impl Detail for Car {
fn brand(&self) -> String {
return self.brand;
}
fn color(&self) -> String {
return self.color;
}
}
After:
impl Detail for Car {
fn brand(&self) -> String {
// using `format` instead of directly returning the brand bcoz it throws error:
// "move occurs because `self.brand` has type `String`, which does not implement the `Copy` trait"
return format!("{}", self.brand);
}
fn color(&self) -> String {
return format!("{}", self.color);
}
}
- Rust by example
- Book: The Rust Programming Language
- Rust for Haskell Programmers!
- Learn Rust Documentation
- Rustlings
- 24 days of Rust
- Learn Rust by aml3
- Rust for C++ programmers
- Learn Rust by KODERHQ
- Learn Rust by Practical Projects
- Learn Rustlings
- Learn Rust by Book via Video
- What is Rust and why is it so popular?
- Understanding the Rust borrow checker
- No auto type deduction for function, but for local variable
- Including Files and Deeply Directories in Rust
- Understand Rust Ownership model by thoughtram
- Memory Safety in Rust: A Case Study with C
- Ownership in Rust by thoughtram
- References in Rust by thoughtram
- Iterators in Rust by thoughtram
- Lifetimes in Rust by thoughram