- Upgrading the Firmware on a Tulip
- Learning Code Through the Advent of Code Challenge
- Common Loki Misconfigurations
- Iterating Through a List in Ink
- Debugging Misconfigured Container Networks
- Minimum Viable EC2 in Terraform
- Storylets in Ink
- Interactive Fiction Tooling Overview
- In-Place Resizing for Digitalocean Droplets
- Unity Demonstrates the Importance of FOSS
- Target Labels in Prometheus
- My View of AI is the Same
- Verify DNS Ownership with TXT Records
- Sane Droplet Defaults
- Editing Made Easy with Vim
- Gatsby Gotchas
- Concatinating Default AWS Tags in Terraform
- Easily Updating the Default Github Branch
- Lifetimes in Rust
- Checking for Bad Links
- Maybe TypeScript and React is Bad
- Static Asset Management in React
- Bundler Down Time
- Using React Context for Localization
- JS Implementation of a Sticky Footer
- Custom Aliases
- Trying Out the 7drl Challenge
- Trash Opinions
- Building Your First Program in Rust
- Fixing mongod reports errors related to opening a socket
- Improving Open Source Maintenance
- Technical Interviewing Tips
- Housekeeping Note
- Dynamic Programming Basics
- The Oddity of Naming Conventions in Programming Languages
- An Experiment Using Machine Learning, Part 3
- Debugging with grep
- An Experiment Using Machine Learning, Part 2
- An Experiment Using Machine Learning, Part 1
- The Value of while
- National Day of Civic Hacking
- OpenAI and the Future of Humanity
- Creating a Whiteboard App in Django
- Creating Meaningful, Organized Information
- Towards A Critique of Social Media Feeds
- Setting up Routes in Django
- Developing a Messaging Component for Code for SF
- Dream Stream 2.0
- Keyed Collections in Javascript: Maps and Sets
- Blog Soft Relaunch
- Scraping with Puppeteer
- Looking Ahead to Dream Stream 2.0
- Solving West of Loathing's Soupstock Lode Puzzle
- Installing Ubuntu
- Interview with David Jickling Evaluation
- Compare Text Evaluation
- Dream Stream Evaluation
Lifetimes in Rust
In Rust, every reference has what is called a lifetime. A lifetime is the period of time when a reference is valid. Like in most programming languages, that is going to be the scope where the reference is assigned. In Rust, it can also go out of scope earlier based on Rust’s borrowing rules.
let x = 5;
fn foo() {
let y = 10;
println!("{}", x);
println!("{}", y);
}
println!("{}", x);
println!("{}", y); // This will raise an error because y does not exist in this scope
Here, y
is defined in the foo
function, and so it is no longer available whenever the function completes its call, so the global space has no knowledge of the y
assignment. (Side note that there is another problem with the section of code, but that has to do with using closures in Rust, which is outside the scope of this blog post).
So far so good. Mostly, lifetimes are the sort of thing you don’t have to think about any more than you have to think about scope in other programming language. Mostly. That is to say, there are cases where you do have to think about them a bit.
In situations where a function has references that the compiler cannot know how long it will need to be valid in advance, it will be necessary to explicitly annotate the lifetime of the function’s references. The simplest example is a function with an if
/else
branch.
fn return_greater<'a>(x: &'a i32, y: &'a i32) -> &'a i32 {
if x > y {
x
} else {
y
}
}
If x
is greater than y
then the references to either don’t need to stick around for the else
block because the function doesn’t need to execute that block. However in cases where y
is greater than x
we do need to at least hold onto the y
reference at that point so it can be returned. The compiler cannot know when it needs to hold onto that reference, and when it is okay to drop itsince the function accepts any valid pair of references to signed 32-bit integers. It’s the entire point of the function.
To address this, we annotate the lifetime, and state that the parameters should exist for the duration of that lifetime. The '
syntax is used for declaring a lifetime. It is idiomatic in Rust to write 'a
for annotated lifetimes. I have no idea why, but you can think of the a
as short for annotated. So at the start of the function, <'a>
is written. That is a lifetime that refers to the the length of the function scope. Then any parameter reference with&'a
preceding it means that that reference should exist at least as long as the function scope. Our function now works. Also note how you only need to use the lifetime syntax within the function signature, not the body.
One other case where lifetime annotations come up is in declaring structs. In addition to having owned data types, structs can also hold references, and in those cases it will need to be annotated. This most commonly comes up with references to string slices.
Lets say we are creating a computer roleplaying game. There are a lot of different types of systems at play, but the one we are interested in implementing is the text block. The text block doesn’t do anything too fancy, it just displays some text to the player, has some metadata associated with it, and has a selection of choices (which are themselves another struct) the player can select to proceed with the story. We can write something like this:
struct TextBlock<'a> {
id: u32,
text: &'a str,
choices: Vec<Choice<'a>>
}
Our Choice struct isn’t defined here, but we can imagine that it also has an annotated lifetime for its own string slice references. When we include that struct in the TextBlock
struct, it too needs to include its annotated lifetime which means that the referenced string slices for Choice
structs need to last at least as long as the scope of a TextBlock
struct. Our Choice
struct lives instead a vector, and so our syntax of Vec<Choice<'a>>
is nearly morphing into the infamous turbofish!
When you are first learning Rust looking at other people’s code can be intimidating because any syntax that is unique to Rust increases the cognitive load. Luckily lifetime annotations are a relatively straightforward concept so knowing how they work is an important piece of the Rust code puzzle.