Learn Rust and Swift in parallel. Comparing the language features.
Day 1: Variables and Mutability
- let defines immutable variables by default
- let x = 1; let x = " "; will create new variable with same name. This is called variable shadowing.
Day 2: Control FLow
- break x + 1; break the loop and return a value
- expression block shall not append ";" to the last line.
Day 3: Setup cargo
- install rust
- install plugin to Visual Studio Code
- terminal: cargo new project to a git folder
- open project folder in VSCode
- build and run
- edit .git/info/exclude, add following lines to ignore non-source files from git repository
- */target/*
- */src/*
- !*/src/*.rs
Day 4: Hands on Rust programming
- ::new is an associated function, i.e. static method
- Result.expect(str) return result(in case Ok()) or str (in case Err(_) catchall)
- match a.cmp(&mut b) {arm1: pattern1 => fn1, arm2: pattern2 => fn2}
- References are immutable by default. Hence &mut b instead of &b
Day 5: Coding Rust with VSCode
- use 'task' to build and run
- Tabnine autocomplete is very handly
Day 6: Ownership 1
- memory is managed through a system of ownership with a set of rules that the compiler checks at compile time. None of the ownership features slow down your program while it’s running.
Day 7: Ownership 2
- There’s a design choice that’s implied by this: Rust will never automatically create “deep” copies of your data.
- Tuples, if they only contain types that also implement Copy. For example, (i32, i32) implements Copy, but (i32, String) does not.
- We cannot have a mutable reference while we have an immutable one.
- It's ok if the last usage of the immutable references occurs before the mutable reference is introduced. A reference’s scope starts from where it is introduced and continues through the last time that reference is used.
fn no_dangle() -> String {
let s = String::from("hello");
s
}
This works without any problems. Ownership is moved out, and nothing is deallocated. 6.slice[..] is of &str.
Something about Trait
- Trait definitions are a way to group method signatures together to define a set of behaviors necessary to accomplish some purpose.
- Use {} for a default implementation
- Default implementations can call other methods in the same trait, even if those other methods don’t have a default implementation.
Day 8 - 9: Structs
- Struct update syntax: ..struct_instant_1
- impl method(&self)
Day 10: Enums and Match
- Enums are a way of grouping related values so you can use them without spelling mistakes.
- you have to convert an Option to a T before you can perform T operations with it.
- The code associated with each arm is an expression, and the resulting value of the expression in the matching arm is the value that gets returned for the entire match expression.
- if let arm = enum {} else {}
- You’ll see this pattern a lot in Rust code: match against an enum, bind a variable to the data inside, and then execute code based on it.
Day 11: Vector
let v = vec![1, 2, 3, 4];
let first = &v[0]; // immutable borrow
v.push(6); // mutable borrow
- you can't have mutable and immutable references in the same scope.
- We can define an enum whose variants will hold the different value types, and then all the enum variants will be considered the same type: that of the enum. Then we can create a vector that holds that enum and so, ultimately, holds different types:
enum SpreadsheetCell {
Int(i32),
Float(f64),
Text(String),
}
let row = vec![
SpreadsheetCell::Int(3),
SpreadsheetCell::Text(String::from("blue")),
SpreadsheetCell::Float(10.12),
];
Day 12: String and HashMap/Dictionary
- Store string in UTF-8
- let s = format!("{}-{}-{}", s1, s2, s3);
Day 13: HashMap
- scores.entry(String::from("Yellow")).or_insert(50);
- ownership:
let field_name = String::from("Favorite color");
let field_value = String::from("Blue");
let mut map = HashMap::new();
map.insert(field_name, field_value);
// field_name and field_value are invalid at this point
- entry().or_insert()
for word in text.split_whitespace() {
let count = map.entry(word).or_insert(0);
*count += 1;
}
Day 14: Error Handling
- two groups: recoverable and unrecoverable
- Result<T, E> for recoverable and panic! for unrecoverable
- When you choose to return a Result value, you give the calling code options rather than making the decision for it.
Day 15: Generics
- handling duplication of concepts.
Day 16: Reference Lifetime
- When returning a reference from a function, the lifetime parameter for the return type needs to match the lifetime parameter for one of the parameters.
Day 17-23: I/O handing
Day 23: Auto-testing
- define test functions which uses assert_eq!(result, function(test_data));
- cargo test -- --show-output
- unit testing cases are defined in the same source file:
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn internal() {
assert_eq!(4, internal_adder(2, 2));
}
}
- integration testing use dedicate files under dedicate folder: "tests".
- use tests/common/mod.rs to store common functions for testing.
Day 24 - 26: Closures
- all ll closures implement at least one of the traits:
- Fn, not changing environment,
- FnMut, changing environment,
- FnOnce, take ownership.
Day 27: Iterator
- Methods that call iterator.next are called consuming adaptors, because calling them uses up the iterator.
- Iterators are one of Rust’s zero-cost abstractions, by which we mean using the abstraction imposes no additional runtime overhead. You can use iterators and closures without fear! They make code seem like it’s higher level but don’t impose a runtime performance penalty for doing so.
Day 28 - 31: Packages and Crates
- A package must contain zero or one library crates, and no more. It can contain as many binary crates as you’d like, but it must contain at least one crate (either library or binary).
- Modelue is private be default. We need to refer to it with "use" to bring items from there into our package’s scope.
- running cargo test will run the code examples in your documentation as tests!
Day 32 - 37: Smart Pointer
- Smart pointers are data structures that not only act like a pointer but also have additional metadata and capabilities.
- reference counting smart pointer type enables you to have multiple owners of data by keeping track of the number of owners and, when no owners remain, cleaning up the data.
- smart pointer from an ordinary struct is that smart pointers implement the Deref and Drop traits
- use Box to store data on the heap
- When the Deref trait is defined for the types involved, Rust will analyze the types and use Deref::deref as many times as necessary to get a reference to match the parameter’s type.
- We can’t disable the automatic insertion of drop when a value goes out of scope, and we can’t call the drop method explicitly. So, if we need to force a value to be cleaned up early, we can use the std::mem::drop function by passing the value we want to force to be dropped early as an argument.
- RefCell allows immutable or mutable borrows checked at runtime.
- Weak_count doesn’t need to be 0 for the Rc instance to be cleaned up.
- Calling the upgrade method on a Weak instance, which will return an Option<Rc>. You’ll get a result of Some if the Rc value has not been dropped yet and a result of None if the Rc value has been dropped. Because upgrade returns an Option<Rc>.
Day 38 - 40: OOP Features
- Inheritance has recently fallen out of favor as a programming design solution in many programming languages because it’s often at risk of sharing more code than necessary. Subclasses shouldn’t always share all characteristics of their parent class but will do so with inheritance. This can make a program’s design less flexible. It also introduces the possibility of calling methods on subclasses that don’t make sense or that cause errors because the methods don’t apply to the subclass. In addition, some languages will only allow a subclass to inherit from one class, further restricting the flexibility of a program’s design.
- Trait objects aren’t as generally useful as objects in other languages: their specific purpose is to allow abstraction across common behavior.
- An object-oriented pattern won’t always be the best way to take advantage of Rust’s strengths, but is an available option.
Day 41 - 42: Patterns and Matching
- Use if let expressions mainly as a shorter way to write the equivalent of a match that only matches one case.
- Using @ lets us test a value and save it in a variable within one pattern.
Day 43: Unsafe
- Unions are primarily used to interface with unions in C code.
Day 44 - 46: Advanced Traits and Types
- Associated types connect a type placeholder with a trait such that the trait method definitions can use these placeholder types in their signatures.
- calling methods with same name: ::baby_name()
- Newtype pattern, which involves creating a new type in a tuple struct which has one field and be a thin wrapper around the type we want to implement a trait for. Then the wrapper type is local to our crate, and we can implement the trait on the wrapper.
- The golden rule of dynamically sized types is that we must always put values of dynamically sized types behind a pointer of some kind.
Day 47: Advanced Funtion Pointer
- Function pointers implement all three of the closure traits (Fn, FnMut, and FnOnce), so you can always pass a function pointer as an argument for a function that expects a closure.
- An example of where you would want to only accept fn and not closures is when interfacing with external code that doesn’t have closures: C functions can accept functions as arguments, but C doesn’t have closures.
Day 48: Macros
- macros are a way of writing code that writes other code, which is known as metaprogramming.
- macro definitions are more complex than function definitions because you’re writing Rust code that writes Rust code. Due to this indirection, macro definitions are generally more difficult to read, understand, and maintain than function definitions.
- you must define macros or bring them into scope before you call them in a file.
Day 49 - 51: Concurrency
- By leveraging ownership and type checking, many concurrency errors are compile-time errors in Rust rather than runtime errors.
- The send function takes ownership of its parameter, and when the value is moved, the receiver takes ownership of it.
- 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 (a reference to T) is Send, meaning the reference can be sent safely to another thread.