- Troels' Newsletter
- Posts
- Understanding References and Borrowing in Rust
Understanding References and Borrowing in Rust
Rust is a language known for its focus on safety and performance, primarily through its innovative use of concepts like ownership and borrowing. For TypeScript developers intrigued by Rust's approach, understanding references is essential. Let's explore how Rust uses this concept to enhance safety, with comparisons to something you might already be familiar with—JavaScript and TypeScript references.
Rust references resemble pointers but come with a layer of safety that ensures the developer doesn't shoot themselves in the foot. In Rust, the concept of ownership plays a significant role, and references are a way to interact with data without taking ownership of it. Imagine Rust as a diligent librarian who doesn't want you to walk out with books under your arm, but is happy to let you read them in the library. This approach lets you manage and use data efficiently without the dread of memory leaks or data races that you might encounter in other low-level languages.
In Rust, borrowing is the process of creating references to a piece of data, allowing multiple function calls to interact with it while guaranteeing that the data itself is not altered or destroyed unexpectedly. This is akin to using const
or referencing an object in TypeScript, where you know your variable won't get overridden out of the blue. By marking your variable as &String
in the function declaration, you're telling Rust, "Hey, I'm just looking at this, no funny business."
Consider the following Rust example where a string object text
is created in the main function and then borrowed by the print
function using a reference:
fn print(s: &String) {
println!("{}", s);
}
fn main() {
let text = String::from("hello");
print(&text); // Borrow `text`
println!("{}", text); // `text` is still usable here
}
In this piece of code, print(&text);
doesn't transfer ownership. This design ensures you can print text
in the print
function and later in the main
function without any issues. Rust's borrow checker ensures that even though you're sharing text
, it remains accessible post-borrow, but no additional modifications occur that could disrupt program integrity.
The story is different in TypeScript. You can pass references directly since JavaScript manages memory automatically, and variables can be reassigned or modified without compiler intervention. In situations where Rust's complexity feels burdensome, you will appreciate the guarantees and protections when handling complex data structures and systems programming.
Rust references eliminate data races by making immutable and mutable references exclusive. Mutable references (&mut
) are like taking a book out for a read with a pencil – Rust makes sure you are the only one doing that, maintaining data integrity.
fn print(s: &String) {
println!("{}", s);
}
fn main() {
let text = String::from("hello");
print(&text); // Borrow text temporarily
println!("{}", text); // Can still use text here
}
In the provided Rust code, we define a function print
that takes a reference to a String
as an argument, s: &String
. This reference does not gain ownership of the data but merely borrows it, allowing s
to be used within print
. The data referenced by s
still resides in main
, and the ownership rules enforced by Rust guarantee that once the function print
is done, the text
can be safely accessed again. This guarantees both safety and efficiency since no new allocation, deallocation, or movement of the value is taking place; instead, you're just pointing and shouting, "Look at that!" from a safe distance without touching.
Here's what's fantastic about this: you can't modify text
through s
because it is not a mutable reference. This setup prevents accidental data modification, a common source of bugs in programming. Since print
doesn't possess ownership, text
remains immutable under the scope of print
, maintaining its original state for further use. In TypeScript, although references provide easier manipulation of objects, it lacks such rigorous compile-time checks, potentially leading to runtime errors, especially in scenarios where shared mutable state is involved.
Moreover, the rigorous checks performed by the Rust compiler prevent dangling references, a common pitfall in languages like C and C++. In Rust's system, references are guaranteed to always point to valid memory locations during their lifetime, ensuring stability across your applications.
Rust's concept of borrowing and references is a powerful feature that enhances code safety by preventing unauthorized access or modification of data. By using references, Rust enables you to work efficiently with your data without needing to duplicate it unnecessarily, much like you would manage references in TypeScript but with additional guarantees. This concept forms a building block for more complex Rust topics, ensuring that your journey from TypeScript to Rust is both informative and secure.
Reply