public/content/blog/2022/rust-changes-how-you-think-and-code.md
eggy cc18627f6c remove albatross
articles are not blog entries
2023-03-25 16:07:40 -04:00

4.7 KiB
Raw Blame History

This article is also published in The FOSS Albatross.

Rust is the hot new language on the block (as new as a language from 2006 can be) that boasts reliability and efficiency.

How does it do this? Well, Rust has something that no other language does — it guarantees memory and thread safety while maintaining the same high performance of C or C++, all the while having high level features such as pattern matching and functional programming!

Some languages come close: Go is known for being both fast to run and to write, but its garbage collector and xenophobia toward other languages adds overhead that means that it is not suited for a systems programming language.

In safe Rust, there is no such thing as undefined behaviour. Everything your code says it does will happen — segfaults and NullPointerExceptions are impossible.

For simpler issues, the rustc compiler tells you more or less exactly what went wrong, along with a helpful error code, a link for examples on how to fix the error code, and even a suggestion that applies directly to your current code, which more times than not immediately fixes the issue.

error: format argument must be a string literal
--> helloworld.rs:3:14
 |
3 | println!(123);
 | ^^^
 |
help: you might be missing a string literal to format with
 |
3 | println!("{}", 123);
 |
error[E0384]: cannot assign twice to immutable variable `a`
 --> helloworld.rs:3:5
 |
2 |   let a = 123;
 |     -
 |     |
 |     first assignment to `a`
 |     help: consider making this binding mutable: `mut a`
3 |   a *= 2;
 |   ^^^^^^ cannot assign twice to immutable variable

Thanks, rustc!

In time, you come to stop thinking less about the edges in the language and focus more on implementing what you want to implement. Rust makes it so that you dont have to stop and ask yourself these questions every minute:

  • What if this variable isnt initialised or defined?
  • What if this variable is already used?
  • What if Im modifying a variable that isnt supposed to be modified?
  • What if another thread changes this data while Im reading it?
  • Did I forget to handle an error?
  • Did I forget to check the error?

So how does it do this?

Immutability by default

There are debates on whether immutable or mutable objects are better. Well, Rust provides both — but you have to explicitly tell Rust that you want your variables to be mutable. For example, the second error message in this article shows that you need the mut keyword to let the compiler let you change variable values.

let mut a = 1;

This applies to everything: from references to function arguments. If a variable isnt passed as mut, its not mutable, and there is nothing else you can do to get around that. This isnt like JavaScripts const, either — the internal variables of a struct also have to be declared mutable in order to overwrite them.

This added friction to mutability means that developers tend to prefer immutable objects when possible, so its very clear when a variable can change!

The borrow checker

Perhaps Rusts flagship feature, this is how Rust manages memory without the complexity of manual memory management or the overhead of a garbage collector. In a nutshell, each variable is given an owner, and they may only have one owner.

You can “borrow” the value if you want to do something with it but give it back to use later, but the typical pitfalls of pointers dont exist in Rust because there are no pointers in Rust! (At least, not safe Rust.)

Heres an example of what Rust prevents — if you operated on vector B, it would change C, so Rusts safety guarantees would not hold. Thats why the compiler doesnt let you run this in the first place.

let a = vec![1, 2, 3];
let b = a;
let c = a;

2 |   let a = vec![1, 2, 3];
 |     - move occurs because `a` has type `Vec<i32>`, which does not implement the `Copy` trait
3 |   let b = a;
 |       - value moved here
4 |   let c = a;
 |       ^ value used here after move

And its here that you really have to appreciate how much information the compiler gives you. It:

  • tells you where the value originated from
  • tells you where the value was used
  • and tells you where the value was used again, which is not allowed

Conclusion

Though there is a rather steep learning curve from just how much there is to unlearn about the finnicky things you can do in more traditional languages, Rust is a language that lets the computer calculate if your program is correct, letting you think purely on how to solve your problems.