rust/src/doc/trpl/variable-bindings.md

170 lines
5.6 KiB
Markdown
Raw Normal View History

2015-01-16 14:30:27 -06:00
% Variable Bindings
Virtually every non-'Hello World Rust program uses *variable bindings*. They
2015-04-09 15:30:43 -05:00
look like this:
2015-04-09 15:30:43 -05:00
```rust
fn main() {
let x = 5;
}
```
2015-04-09 15:30:43 -05:00
Putting `fn main() {` in each example is a bit tedious, so well leave that out
in the future. If youre following along, make sure to edit your `main()`
function, rather than leaving it off. Otherwise, youll get an error.
2015-04-09 15:30:43 -05:00
In many languages, this is called a *variable*, but Rusts variable bindings
have a few tricks up their sleeves. For example the left-hand side of a `let`
expression is a [pattern][pattern], not just a variable name. This means we
can do things like:
2015-04-09 15:30:43 -05:00
```rust
let (x, y) = (1, 2);
```
After this expression is evaluated, `x` will be one, and `y` will be two.
2015-04-09 15:30:43 -05:00
Patterns are really powerful, and have [their own section][pattern] in the
book. We dont need those features for now, so well just keep this in the back
of our minds as we go forward.
[pattern]: patterns.html
Rust is a statically typed language, which means that we specify our types up
2015-04-09 15:30:43 -05:00
front, and theyre checked at compile time. So why does our first example
compile? Well, Rust has this thing called type inference. If it can figure
out what the type of something is, Rust doesnt require you to actually type it
out.
We can add the type if we want to, though. Types come after a colon (`:`):
2015-04-09 15:30:43 -05:00
```rust
let x: i32 = 5;
```
2015-04-09 15:30:43 -05:00
If I asked you to read this out loud to the rest of the class, youd say “`x`
is a binding with the type `i32` and the value `five`.”
In this case we chose to represent `x` as a 32-bit signed integer. Rust has
many different primitive integer types. They begin with `i` for signed integers
and `u` for unsigned integers. The possible integer sizes are 8, 16, 32, and 64
bits.
In future examples, we may annotate the type in a comment. The examples will
look like this:
2015-04-09 15:30:43 -05:00
```rust
fn main() {
let x = 5; // x: i32
}
```
2015-04-09 15:30:43 -05:00
Note the similarities between this annotation and the syntax you use with
`let`. Including these kinds of comments is not idiomatic Rust, but we'll
occasionally include them to help you understand what the types that Rust
infers are.
By default, bindings are *immutable*. This code will not compile:
2015-04-09 15:30:43 -05:00
```rust,ignore
let x = 5;
x = 10;
```
It will give you this error:
```text
error: re-assignment of immutable variable `x`
x = 10;
^~~~~~~
```
If you want a binding to be mutable, you can use `mut`:
2015-04-09 15:30:43 -05:00
```rust
let mut x = 5; // mut x: i32
x = 10;
```
There is no single reason that bindings are immutable by default, but we can
2015-04-09 15:30:43 -05:00
think about it through one of Rusts primary focuses: safety. If you forget to
say `mut`, the compiler will catch it, and let you know that you have mutated
something you may not have intended to mutate. If bindings were mutable by
default, the compiler would not be able to tell you this. If you _did_ intend
mutation, then the solution is quite easy: add `mut`.
2015-04-09 15:30:43 -05:00
There are other good reasons to avoid mutable state when possible, but theyre
out of the scope of this guide. In general, you can often avoid explicit
mutation, and so it is preferable in Rust. That said, sometimes, mutation is
2015-04-09 15:30:43 -05:00
what you need, so its not verboten.
2015-04-09 15:30:43 -05:00
Lets get back to bindings. Rust variable bindings have one more aspect that
differs from other languages: bindings are required to be initialized with a
value before you're allowed to use them.
2015-04-09 15:30:43 -05:00
Lets try it out. Change your `src/main.rs` file to look like this:
2015-04-09 15:30:43 -05:00
```rust
fn main() {
let x: i32;
println!("Hello world!");
}
```
2015-04-09 15:30:43 -05:00
You can use `cargo build` on the command line to build it. Youll get a
warning, but it will still print "Hello, world!":
```text
Compiling hello_world v0.0.1 (file:///home/you/projects/hello_world)
2015-04-09 15:30:43 -05:00
src/main.rs:2:9: 2:10 warning: unused variable: `x`, #[warn(unused_variable)]
on by default
src/main.rs:2 let x: i32;
^
```
2015-04-09 15:30:43 -05:00
Rust warns us that we never use the variable binding, but since we never use
it, no harm, no foul. Things change if we try to actually use this `x`,
however. Lets do that. Change your program to look like this:
2015-04-09 15:30:43 -05:00
```rust,ignore
fn main() {
let x: i32;
println!("The value of x is: {}", x);
}
```
2015-04-09 15:30:43 -05:00
And try to build it. Youll get an error:
2015-04-09 15:30:43 -05:00
```bash
$ cargo build
Compiling hello_world v0.0.1 (file:///home/you/projects/hello_world)
src/main.rs:4:39: 4:40 error: use of possibly uninitialized variable: `x`
src/main.rs:4 println!("The value of x is: {}", x);
^
note: in expansion of format_args!
<std macros>:2:23: 2:77 note: expansion site
<std macros>:1:1: 3:2 note: in expansion of println!
src/main.rs:4:5: 4:42 note: expansion site
error: aborting due to previous error
Could not compile `hello_world`.
```
2015-04-09 15:30:43 -05:00
Rust will not let us use a value that has not been initialized. Next, lets
talk about this stuff we've added to `println!`.
If you include two curly braces (`{}`, some call them moustaches...) in your
string to print, Rust will interpret this as a request to interpolate some sort
of value. *String interpolation* is a computer science term that means "stick
in the middle of a string." We add a comma, and then `x`, to indicate that we
2015-04-09 15:30:43 -05:00
want `x` to be the value were interpolating. The comma is used to separate
arguments we pass to functions and macros, if youre passing more than one.
When you just use the curly braces, Rust will attempt to display the value in a
meaningful way by checking out its type. If you want to specify the format in a
more detailed manner, there are a [wide number of options available][format].
For now, we'll just stick to the default: integers aren't very complicated to
print.
[format]: ../std/fmt/index.html