& vs. ref in Rust patterns
Posted on Thu 02 June 2016 in Code • Tagged with Rust, pattern matching, borrowing, references • Leave a comment
Rust is one of those nice languages with pattern matching.
If you don’t know, it can be thought of as a generalization of the switch
statement:
comparing objects not just by value (or overloaded equality operator, etc.)
but by structure:
match hashmap.get(&key) {
Some(value) => do_something_with(value),
None => { panic!("Oh noes!"); },
}
It doesn’t end here. As you can see above, objects can also be
destructured during the match (Some(value)
),
their parts assigned to bindings (value
),
and those bindings can subsequently be used in the match
branch.
Neat? Definitely.
In Rust, pattern matching is bread-and-butter of not only the match
statement,
but also for
, (if
)let
, and even ordinary function arguments.
Mixing in Rust semantics
For a long time, however, I was somewhat confused as to what happens
when references and borrowing is involved in matching.
The two “operators” that often occur there are &
(ampersand) and ref
.
You should readily recognize the first one, as it is used pervasively in Rust to create references (and reference types). The second one quite obviously hints towards references as well. Yet those two constructs serve very different purposes when used within a pattern.
To add to the confusion, they are quite often encountered together:
use hyper::Url;
// print query string params of some URL
let url = Url::parse(some_url).unwrap();
let query_params: Vec<(String, String)> = url.query_pairs().unwrap_or(vec![]);
for &(ref name, ref value) in &query_params {
println!("{}={}", name, value);
}
Lack of one or the other will be (at best) pointed out to you by the compiler, along with a suggestion where to add it. But addressing problems in this manner can only go so far. So how about we delve deeper and see what it’s really about?
Part of the reference, part of the pattern
Rust is very flexible as to what value can be a subject of pattern matching.
You would be very hard pressed to find anything that cannot be used within a match
statement, really.
Both actual objects and references to objects are perfectly allowed:
struct Foo(i32);
// ...
let foo = &Foo(42);
match foo {
x => println!("Matched!"),
}
In the latter case, however, we aren’t typically interested in the reference itself (like above). Instead, we want to determine some facts about the object it points to:
match foo {
&Foo(num) => println!("Matched with number {}", num),
}
As you can see, this is where the ampersand comes into play.
Just like a type constructor (Some
, Ok
, or Foo
above),
the &
operator informs the Rust compiler what kind of value we’re expecting from the match.
When it sees the ampersand, it knows we’re looking for references to certain objects,
and not for the objects themselves.
Why is the distinction between an object and its reference important, though? In many other places, Rust is perfectly happy to blur the gap between references and actual objects1 — for example when calling most of their methods.
Pattern matching, however,
due to its ability to unpack values into their constituent parts,
is a destructive operation. Anything we apply match
(or similar construct) to
will be moved into the block by default:
let maybe_name = Some(String::from("Alice"));
// ...
match maybe_name {
Some(n) => println!("Hello, {}", n),
_ => {},
}
do_something_with(maybe_name)
Following the typical ownership semantics, this will prevent any subsequent moves and essentially consume the value:
error: use of partially moved value: `maybe_name` [E0382]
do_something_with(maybe_name);
^~~~~~~~~~
So just like the aforementioned type constructors (Some
, etc.),
the ampersand operator is simply part of the pattern that we match against.
And just like with Some
and friends, there is an obvious symmetry here:
if &
was used to create the value, it needs to be used when unpacking it.
The syntax used in a pattern that destructures an object is analogous to one used by the expression which created it.
Preventing the move
Errors like the one above often contain helpful notes:
note: `(maybe_name:core::option::Option::Some).0` moved here because it has type `collections::string::String`, which is moved by default
Some(n) => println!("Hello, {}", n),
^
as well as hints for resolving them:
help: if you would like to borrow the value instead, use a `ref` binding as shown:
Some(ref n) => println!("Hello, {}", n),
Here’s where ref
enters the scene.
The message tells us that if we add a ref
keyword in the suggested spot,
we will switch from moving to borrowing for the match
binding that follows (here, n
).
It will still capture its value exactly as before,
but it will no longer assume ownership of it.
This is the crucial difference.
Unlike the ampersand, ref
is not something we match against.
It doesn’t affect what values match the pattern it’s in, and what values don’t2.
The only thing it changes is how parts of the matched value are captured by the pattern’s bindings:
- by default, without
ref
, they are moved into thematch
arms - with
ref
, they are borrowed instead and represented as references
Looking at our example, the n
binding in Some(n)
is of type String
:
the actual field type from the matched structure.
By contrast, the other n
in Some(ref n)
is a &String
—
that is, a reference to the field.
One is a move, the other one is a borrow.
ref
annotates pattern bindings to make them borrow rather than move. It is not a part of the pattern as far as matching is concerned.
Used together
To finish off, let’s untangle the confusing example from the beginning of this post:
for &(ref name, ref value) in &query_params {
println!("{}={}", name, value);
}
Since we know ref
doesn’t affect whether or not the pattern matches,
we could just as well have something like &(a, b)
.
And this should be quite a bit easier to read:
it clearly denotes we expect a reference to a 2-tuple of simple objects.
Not coincidentally,
such tuples are items from the vector we’re iterating over.
Problem is,
without the ref
s we will attempt to move those items into the loop scope.
But due to the way the vector is iterated over (&query_params
),
we’re only borrowing each item, so this is actually impossible.
In fact, it would be a classic attempt to move out of a borrowed context.
It is also wholly unnecessary. The only thing this loop does is printing the items out, so accessing them through references is perfectly fine.
And this is exactly what the ref
operator gives us.
Adding the keyword back,
we will switch from moving the values to just borrowing them instead.
To sum up
-
&
denotes that your pattern expects a reference to an object. Hence&
is a part of said pattern:&Foo
matches different objects thanFoo
does. -
ref
indicates that you want a reference to an unpacked value. It is not matched against:Foo(ref foo)
matches the same objects asFoo(foo)
.
-
The technical term for this is a
Deref
coercion. ↩ -
We can say that it doesn’t affect the satisfiability (or conversely, refutability) of the pattern. ↩