A curated list of Rust interview preparation materials for busy software engineers. Submit a PR if you know of any other Rust interview questions.
🏆 Send me a PR if you know of any Rust interview questions
Rust is a modern computer programming language developed by Mozilla in 2010. It was initially developed to tackle the issue of invalid memory access faced by C and C++ developers while building applications.
Rust is a statically compiled language with a rich type system and ownership model. It can be used to power performance-critical services while guaranteeing memory safety and thread safety, empowering developers to debug at compile-time. In addition to that Rust has great documentation and a user-friendly compiler with top-end tools like integrated package managers and multi-editor with features like type inspection and auto-completion.
Rust prevents all crashes, and, interestingly, Rust is safe by default like JavaScript, Ruby, and Python. This is much more powerful than C/C++ because we cannot write the wrong parallel code you can never see fault in rust. It is very fast in representing a lot of programming paradigms very well.
-
What is Rust?
Rust is a systems programming language that focuses on safety, concurrency, and performance. It was developed by Mozilla Research and first appeared in 2010. Rust aims to address common pitfalls in programming, such as memory safety issues, data races, and undefined behavior, often found in languages like C and C++.
Here are some key features and concepts of Rust:
-
Memory Safety: Rust's most distinctive feature is its system, which enables fine-grained control over memory allocation and deallocation. It ensures memory safety without relying on garbage collection. The ownership system prevents common issues like null pointer dereferences, dangling pointers, and data races.
-
Concurrency: Rust provides powerful concurrency primitives, including threads, message passing, and shared-state concurrency. It enforces strict rules at compile-time to prevent data races and ensures thread safety.
-
Performance: Rust is designed for high-performance systems programming. It provides control over low-level details while ensuring safety. Its zero-cost abstractions allow high-level code to have the same performance as equivalent low-level code.
-
Expressive Type System: Rust has a strong and expressive type system. It includes features like pattern matching, algebraic data types, and generics, allowing developers to write generic and reusable code. The type inference system reduces the need for explicit type annotations.
-
Error Handling: Rust has a robust error handling mechanism based on the "Result" type and the "Option" type. This approach encourages developers to handle errors explicitly and helps avoid unexpected runtime errors.
-
Cargo: Rust's package manager and build system, called Cargo, simplifies dependency management and building projects. It provides tools for testing, documentation generation, and benchmarking, making it easier to develop and maintain Rust projects.
-
Community and Ecosystem: Rust has a vibrant and growing community. It has an extensive ecosystem of libraries and frameworks, known as "crates," which cover a wide range of domains. The Rust community is known for its helpfulness and focus on education.
-
-
What is the cargo and what are the most popular cargo commands?
Cargo is the Rust package manager that can download dependencies, compile the project, make distributable packages, and upload them to the Rust community's package registry creates.io. The most popular cargo commands are:
- Create a new package that builds an executable:
cargo new IntmainApp
- Create a new package that builds a library:
cargo new --lib IntmainLib
- Build a package and all its dependencies:
cargo build
- Build a package with optimization:
cargo build -- release
- Run a binary of local package:
cargo run
. Arguments to the binary are followed by two dashes (--):cargo run -- args
- Remove generated artifacts from the target directory:
cargo clean
- Display a tree of dependencies in the project:
cargo tree
- Compile and execute unit, integration, and documentation tests:
cargo test
- Create a new package that builds an executable:
-
What is Cargo.toml and Cargo.lock?
The Cargo.toml file is a manifest file where we can specify metadata such as name, version, etc, project settings, and dependencies(crates) for our project. The Cargo.lock file contains exact information about dependencies and Cargo maintains it. If we are building a rust library, then put Cargo.lock in the .gitignore but if we are building end products application then Cargo.lock should be checked into the git repo.
-
Difference between String and str in Rust?
The
String
type is the most common string type in rust and it has ownership over the content of the string. It is heap-allocated, growable, and not null-terminated. A String is stored as a vector of bytes(Vec<u8>
), and guaranteed to be a valid UTF-8 sequence. String literals orstr
is the most primitive string type. A Rust &str is like achar*
.str
is an immutable sequence of UTF-8 bytes of dynamic length somewhere in memory. Since the size is unknown, one can only handle it behind a pointer, hence it most commonly appears as&str
: a reference to some UTF-8 data, normally called a "string slice" -
Copy vs Clone in Rust?
The
Clone
trait defines the ability to explicitly create a deep copy of an object T. When we callClone
for type T, it does all the arbitrarily complicated operations required to create a new T. TheCopy
trait in Rust defines the ability to implicitly copy an object. The behaviorCopy
is not overloadable. It is always a simple bitwise copy. This is available for the types that have a fixed size and are stored entirely on the stack. Read more about copy and clone traits here... -
Struct in rust?
Struct is a user-defined data type that can hold different types of data together.
struct
the keyword is used to create structure in Rust. Example of struct://Defining a struct struct Employee{ empid :String, name :String, email :String, } //Instantiating a struct let emp = Employee{empid: 5, name: "Dev", email: "dev@intmain.co"}; **//Accessing the Struct data** let id = emp.empid; let name = emp.name; let email = emp.email;
-
Why do we use the owned String type rather than the &str string slice type as the struct field?
Each instance of a struct owns all of its data and for that, the data should remain valid as long as the struct is valid. Hence we use owned string type rather than the string slice. It’s also possible for structs to store references to data owned by something else, but to do so we are required to use the lifetimes because lifetimes ensure that the data referenced by a struct is valid for as long as the struct is.
-
What is the Option type in Rust, and why is it useful?
The
Option
type in rust is used to represent an optional value(presence or absence of a value). Every option is eitherSome
and contains some value orNone
and doesn't contain any value. To check if the Option type contains some value so that you can unwrap it and access it, use theis_some()
method which returns true if the option type is some. -
What is a trait in Rust, and how is it different from an interface in other languages?
The Rust traits are a set of methods that can be defined for particular types. Traits allow you to define shared behavior that can be used by multiple types. A trait method is able to access other methods within that trait.
Let's say we want to create a library crate that can display summaries of data that might be stored in a NewsArcticle, BlogPost struct instance. To do this, we need a summary from each type, and we’ll request that summary by calling a
summarize
method on an instance.pub trait Summary{ fn summarize(&self) -> String { format!("Read More {}, by {} ...", self.title, self.author) } fn summarize_post(&self) -> String; } struct NewsArticle{ title :String, author :String, content :String, } impl Summary for NewsArticle{ fn summarize_post(&self) -> String{ format!("{}", self.content); } } struct BlogPost{ title :String, author :String, post :String, } impl Summary for BlogPost{ fn summarize_post(&self) -> String{ format!("{}", self.content); } } let news = NewsArticle { title : String::from("News Title"), author : String::from("Author name"), content: String::from("Latest news in detail", }; println!("News summary: {}", news.summarize()); println!("Post summary: {}", news.summarize\_post());
-
What is unsafe in rust and when to use it?
Rust guarantees memory safety and it is enforced at runtime, but Rust also provides functionality for bypassing these safety checks and this can be done by placing the code in
unsafe
block. Unsafe code tells the compiler, “Trust me, I know what I’m doing.”unsafe
keyword is used to mark blocks, functions, or traits that contain code that the compiler cannot guarantee to be safe. Unsafe provides the ability to:- Dereference a raw pointer.
- Call an unsafe function or method.
- Access or modify a mutable static variable.
- Implement an unsafe trait.
- Access fields of
union
Remember that unsafe doesn’t turn off the borrow checker or disable any other of Rust’s safety checks: if you use a reference in unsafe code, it will still be checked.