rust-book-cn/nostarch/chapter3.md

97 KiB
Raw Blame History

[TOC]

Up and Running

Well start our Rust journey by talking about the absolute basics, concepts that appear in almost every programming language. Many programming languages have much in common at their core. None of the concepts presented in this chapter are unique to Rust, but well discuss Rusts particular syntax and conventions concerning these common concepts.

Specifically, well be talking about variable bindings, functions, basic types, comments, if statements, and looping. These foundations will be in every Rust program, and learning them early will give you a strong core to start from.

Anatomy of a Rust Program

The foundation of virtually every program is the ability to store and modify data, but to create this data, you first have to create a program. Here, we'll write some code that demonstrates how to begin a Rust program, how to bind a variable, and how to print text to the terminal.

A Simple Program that Binds a Variable

Lets start with a short example that binds a value to a variable, and then uses that binding in a sentence that we'll print to the screen. First, well generate a new project with Cargo. Open a terminal, and navigate to the directory you want to store your projects in. From there, generate a new project:

$ cargo new --bin bindings
$ cd bindings

This creates a new project called bindings and sets up our Cargo.toml and src/main.rs files. As we saw in Chapter XX, Cargo will generate these files and create a little "hello world" program like this:

fn main() {
    println!("Hello, world!");
}

Open that program and replace its code with the following:

fn main() {
    let x = 5;

    println!("The value of x is: {}", x);
}

This is the full program for our example. Enter the run command now to to see it working:

$ cargo run
   Compiling bindings v0.1.0 (file:///projects/bindings)
     Running `target/debug/bindings`
The value of x is: 5

If you get an error instead of this output, double check that you've copied the program exactly as written, and then try again. Now lets break this program down, line by line.

Starting a Program with the main() Function

Most Rust programs open with the same first line as this one from our example program:

fn main() {

The main() function is the entry point of every Rust program. It doesnt have to be at the very beginning of our source code, but it will be the first bit of code that runs when we execute our program. Well talk more about functions in the next section, but for now, just know is that main() is where our program begins. The opening curly brace ({) indicates the start of the functions body.

Binding a Variable with let

Inside the function body, we added the following:

    let x = 5;

This is a let statement, and it binds the value 5 to the variable x. Basic let statements take the following form:

let NAME = EXPRESSION;

A let statement first evaluates the EXPRESSION, and then binds the resulting value to NAME to give us a variable to use later in the program. Notice the semicolon at the end of the statement, too. As in many other programming languages, statements in Rust must end with a semicolon.

In this simple example, the expression already is a value, but we could achieve the same result like this:

let x = 2 + 3;

The expression 2 + 3 would evaluate to 5, which would in turn be stored in the x variable binding. In general, let statements work with patterns. Patterns are part of a feature of Rust called pattern matching. We can compare an expression against a pattern, and then make a choice based on how the two compare. A name like x is a particularly humble form of pattern; it will always match. Patterns are a big part of Rust, and well see more complex and powerful patterns as we go along.

Printing to the Screen with a Macro

The next line of our program is:

    println!("The value of x is: {}", x);

The println! command is a macro that prints the text passed to it to the screen. Macros are indicated with the ! character. In Chapter , you'll learn how to write macros your, but for now we'll use macros provided by the standard Rust library. Macros can add new syntax to the language, and the ! is a reminder that things may look slightly unusual.

The println! macro only requires one argument: a format string. You can add optional arguments inside this format string by using the special text {}. Each instance of {} corresponds to an additional argument. Heres an example:

let x = 2 + 3;
let y = x + 5;

println!("The value of x is {}, and the value of y is {}", x, y);

If you were to run a program containing these statements, it would print the following:

The value of x is 5, and the value of y is 10

Think of {} as little crab pincers, holding a value in place. The first {} holds the first value after the format string, the second set holds the second value, and so on.

The {} placeholder has a number of more advanced formatting options that well discuss later.

After the println macro, we match the opening curly brace that declared the main() function with a closing curly brace to declare the end of the function:

}

And of course, when we run the program, our output is:


The value of x is: 5

With this simple program, you've created your first variable and used your first Rust macro. That makes you a Rust programmer. Welcome! Now that you've seen the basics, let's explore variable bindings further.

Variable Bindings in Detail

So far, weve created the simplest kind of variable binding, but the let statement has some tricks up its sleeve. Lets do some more complex things: create multiple bindings at once, how to add type annotations, mutating bindings, shadowing, and more.

Creating Multiple Bindings

The previous example program just bound one variable, but it's also possible to create multiple variable bindings in one go. Lets try a more complex example, creating two variable bindings at once. Change your example program to this:

fn main() {
    let (x, y) = (5, 6);

    println!("The value of x is: {}", x);
    println!("The value of y is: {}", y);
}

And enter cargo run to run it:

$ cargo run
   Compiling bindings v0.1.0 (file:///projects/bindings)
     Running `target/debug/bindings`
The value of x is: 5
The value of y is: 6

Weve created two bindings with one let statement!

The let statement binds the values in (5, 6) to the corresponding patterns of (x, y). The first value 5 binds to the first part of the pattern, x, and the second value 6 binds to y. We could alternatively have used two let statements to the same effect, as follows:

fn main() {
    let x = 5;
    let y = 6;
}

In simple cases like this, where we are only binding two variables, two let statements may be clearer in the code, but when you're creating many multiple bindings, it's useful to be able to do so all at once. Deciding which technique to use is mostly a judgement call, and as you become more proficient in Rust, youll be able to figure out which style is better in each case.

Delayed Initialization

The examples so far have all provided bindings with an initial value, but that isn't always necessary. Rather, we can assign a value for the binding later, after the let statement. To try this out, write the following program:

fn main() {
    let x;

    x = 5;

    println!("The value of x is: {}", x);
}

And enter cargo run to run it:

$ cargo run
   Compiling bindings v0.1.0 (file:///projects/bindings)
     Running `target/debug/bindings`
The value of x is: 5

As you can see, this works just like the previous program, in which we assigned an initial value.

This raises an interesting question: what happens if we try to print out a binding before we declare a value? Let's find out. Modify your code to look like the following:

fn main() {
    let x;

    println!("The value of x is: {}", x);

    x = 5;
}

When you enter cargo run to run this code, you should see output like this after the command:

   Compiling bindings v0.1.0 (file:///projects/bindings)
src/main.rs:4:39: 4:40 error: use of possibly uninitialized variable: `x` [E0381]
src/main.rs:4     println!("The value of x is: {}", x);
                                                    ^
<std macros>:2:25: 2:56 note: in this expansion of format_args!
<std macros>:3:1: 3:54 note: in this expansion of print! (defined in <std macros>)
src/main.rs:4:5: 4:42 note: in this expansion of println! (defined in <std macros>)
src/main.rs:4:39: 4:40 help: run `rustc --explain E0381` to see a detailed explanation
error: aborting due to previous error
Could not compile `bindings`.

To learn more, run the command again with --verbose.

There's been an error! The compiler wont let us write a program like this, and instead it requests that you declare a value. This is our first example of the compiler helping us find an error in our program. Different programming languages have different ways of approaching this problem. Some languages will always initialize values with some sort of default. Other languages leave the value uninitialized, and make no promises about what happens if you try to use something before initialization. Rust responds with an error to prod the programmer to declare the value they want. We must initialize any variable before we can use it.

Extended Error Explanations

Now that you've seen an example of a Rust error, I want to point out one particularly useful aspect of errors. Rust encourages you to seek further information on the kind of error you've received with output like this:

src/main.rs:4:39: 4:40 help: run `rustc --explain E0381` to see a detailed explanation

This tells us that if we pass the --explain flag to rustc with the provided error code, we can see an extended explanation, which will try to explain common causes of and solutions to that kind of error. Not every error has a longer explanation, but many do. Heres the explanation for the E0381 error we received previously:

$ rustc --explain E0381
It is not allowed to use or capture an uninitialized variable. For example:

fn main() {
    let x: i32;

    let y = x; // error, use of possibly uninitialized variable

To fix this, ensure that any declared variables are initialized before being
used.

These explanations can really help if youre stuck on an error, so don't hesitate to look up the error code. The compiler is your friend, and it's there to help.

Mutable bindings

By default, variable bindings are immutable, meaning that once a value is bound, you can't change that value. Try writing the following sample program to illustrate this:

fn main() {
    let x = 5;

    x = 6;

    println!("The value of x is: {}", x);
}

Save and run the program, and you should receive another error message, as in this output:

$ cargo run
   Compiling bindings v0.1.0 (file:///projects/bindings)
src/main.rs:4:5: 4:10 error: re-assignment of immutable variable `x` [E0384]
src/main.rs:4     x = 6;
                  ^~~~~
src/main.rs:4:5: 4:10 help: run `rustc --explain E0384` to see a detailed explanation
src/main.rs:2:9: 2:10 note: prior assignment occurs here
src/main.rs:2     let x = 5;
                      ^

The error includes the message re-assigment of immutable variable because the program tried to assign a second value to the x variable. But bindings are immutable only by default; you can make them mutable by adding mut in front of the variable name. For example, change the program you just wrote to the following:

fn main() {
    let mut x = 5;

    println!("The value of x is: {}", x);

    x = 6;

    println!("The value of x is: {}", x);
}

Running this, we get:

$ cargo run
   Compiling bindings v0.1.0 (file:///projects/bindings)
     Running `target/debug/bindings`
The value of x is: 5
The value of x is: 6

Using mut, we change the value that x binds to from 5 to 6. Note, however, that mut is part of the pattern in the let statement. This becomes more obvious if we add mutability to a pattern that binds multiple variables, like this:

fn main() {
    let (mut x, y) = (5, 6);

    x = 7;
    y = 8;
}

If you run this code, the compiler will output an error:

$ cargo build
   Compiling bindings v0.1.0 (file:///projects/bindings)
src/main.rs:5:5: 5:10 error: re-assignment of immutable variable `y` [E0384]
src/main.rs:5     y = 8;
                  ^~~~~
src/main.rs:5:5: 5:10 help: run `rustc --explain E0384` to see a detailed explanation
src/main.rs:2:17: 2:18 note: prior assignment occurs here
src/main.rs:2     let (mut x, y) = (5, 6);
                              ^

The way mut is used here, the compiler is fine with reassigning the x variable, but not the y variable. That's because mut only applies to the name that directly follows it, not the whole pattern. For the compiler to allow you to reassign the y variable, you'd need to write the pattern as (mut x, mut y) instead.

One thing to know about mutating bindings: mut allows you to mutate the binding, but not what the name binds to. In other words, the value is not what changes, but rather the path between the value and the name. For example:

fn main() {
    let mut x = 5;

    x = 6;
}

This does not change the value that x is bound to, but creates a new value (6) and changes the binding so that it binds the name x to this new value instead. This subtle but important difference will become more important as your Rust programs get more complex.

Variable Binding Scope

Another important thing to know about variable bindings is that they are only valid as long as they are in scope. That scope begins at the point where the binding is declared, and ends at with the curley brace that closes the block of code containing it. We cannot access bindings "before they come into scope" or "after they go out of scope." Heres an example to illustrate this:

fn main() {
    println!("x is not yet in scope");

    let x = 5;

    println!("x is now in scope");

    println!("In real code, wed now do a bunch of work."); 

    println!("x will go out of scope now! The next curly brace is ending the main function.");
}

The variable binding for x goes out of scope with the last curly brace in the main() function.

This example only has one scope, though. In Rust, it's possible to create arbitrary scopes within a scope by placing code within another pair of curly braces. For example:


fn main() {

    println!("x is not yet in scope");

    let x = 5;

    println!("x is now in scope");

    println!("Lets start a new scope!");

    {
        let y = 5;

        println!("y is now in scope");
        println!("x is also still in scope");

        println!("y will go out of scope now!");
        println!("The next curly brace is ending the scope we started.");
    }

    println!("x is still in scope, but y is now out of scope and is not usable");

    println!("x will go out of scope now! The next curly brace is ending the main function.");
}

The y variable is only in scope in the section of the code that's between the nested pair of curly braces, whereas x is in scope from the let statement that binds it until the final curly brace. The scope of bindings will become much more important later, as you learn about references in Chapter XX.

Shadowing Earlier Bindings

One final thing about bindings: they can shadow previous bindings with the same name. Shadowing is what happens when you declare two bindings with the same name, we say that the second binding shadows the first.

This can be useful if youd like to perform a few transformations on a value, but still leave the binding immutable. For example:

fn main() {
    let x = 5;

    let x = x + 1;

    let x = x * 2;

    println!("The value of x is: {}", x);
}

This program first binds x to a value of 5. Then, it shadows x, taking the original value and adding 1 so that the value of x is then 6. The third let statement shadows x again, taking the previous value and multiplying it by 2 to give x a final value of 12. If you run this, it will output:

$ cargo run
   Compiling bindings v0.1.0 (file:///projects/bindings)
     Running `target/debug/bindings`
The value of x is: 12

Shadowing is useful because it lets us modify x without having to make the variable mutable. This means the compiler will still warn us if we accidentally try to mutate x directly later. For example, say after calculating 12 we dont want x to be modified again; if we write the program in a mutable style, like this:

fn main() {
    let mut x = 5;

    x = x + 1;
    x = x * 2;

    println!("The value of x is: {}", x);

    x = 15;

    println!("The value of x is: {}", x);
}

Rust is happy to let us mutate x again, to 15. A similar program using the default immutable style, however, will let us know about that accidental mutation. Here's an example:

fn main() {
    let x = 5;
    let x = x + 1;
    let x = x * 2;

    println!("The value of x is: {}", x);

    x = 15;

    println!("The value of x is: {}", x);
}

If we try to compile this, we get an error:

$ cargo build
   Compiling bindings v0.1.0 (file:///projects/bindings)
src/main.rs:8:5: 8:11 error: re-assignment of immutable variable `x` [E0384]
src/main.rs:8     x = 15;
                  ^~~~~~
src/main.rs:8:5: 8:11 help: run `rustc --explain E0384` to see a detailed explanation
src/main.rs:4:9: 4:10 note: prior assignment occurs here
src/main.rs:4     let x = x * 2;
                      ^
error: aborting due to previous error
Could not compile `bindings`.

Since we don't want the binding to be mutable, this exactly what should happen.

Shadowing Over Bindings

You can also shadow bindings over one another, without re-using the initial binding. Here's how that looks:


fn main() {
    let x = 5;
    let x = 6;

    println!("The value of x is: {}", x);
}  

Running this sample program, we can see the shadowing in action:

$ cargo run
   Compiling bindings v0.1.0 (file:///projects/bindings)
src/main.rs:2:9: 2:10 warning: unused variable: `x`, #[warn(unused_variables)] on by default
src/main.rs:2     let x = 5;
                      ^
     Running `target/debug/bindings`
The value of x is: 6

Rust gives the value of x as 6, which is the value from the second let statement. There are a few interesting things in this output. First, that Rust will compile and run the program without issue. This is because we haven't mutated the value; instead, we declared a new binding that is also named x, and gave it a new value.

The other interesting thing in this output is this error line:

src/main.rs:2:9: 2:10 warning: unused variable: `x`, #[warn(unused_variables)] on by default

Rust is pointing out that we shadowed x, but never used the initial value. Doing so isnt wrong, but Rust is checking whether this is intentional and not just a mistake. In this case, the compiler issues a warning, but still compiles our program. A warning like this is called a lint, which is an old term for the bits of fluff and fibers in sheeps wool that you wouldn't want to put in cloth.

Similarly, this lint is telling us that we may have an extra bit of code (the statement let x = 5) that we dont need. Even though our program works just fine, listening to these warnings and fixing the problems they point out is worthwhile, as they can be signs of a larger problem. In this case, we may not have realized that we were shadowing x, when we meant to, say, define a new variable with a different name.

Shadowing can take some time to get used to, but its very powerful, and works well with immutability.

Shadowing and Scopes

Like any binding, a binding that shadows another binding becomes invalid at the end of a scope. Heres an example program to illustrate this:

fn main() {
    let x = 5;

    println!("Before shadowing, x is: {}", x);

    {
        let x = 6;

        println!("Now that x is shadowed, x is: {}", x);
    }

    println!("After shadowing, x is: {}", x);
}

This code first creates the x variable and prints x to the terminal. Then, inside a new scope, it creates a new binding for x with a new value, and prints that value. When the arbitrary scope ends, x is printed once more. If we run this example, we can see the shadow appear and disappear in the output:

$ cargo run
   Compiling bindings v0.1.0 (file:///projects/bindings)
     Running `target/debug/bindings`
Before shadowing, x is: 5
Now that x is shadowed, x is: 6
After shadowing, x is: 5

In this case, the binding value reverts to the original value once the shadow binding goes out of scope.

How Functions Work in Rust

Functions are pervasive in Rust code. Weve already seen one of the most important functions in the language: the main() function thats the start of every program. We've also seen the fn keyword, which allows us to declare new functions.

Rust code uses snake case as the conventional style for function names. In snake case, all letters are lower case, and there are underscores separating words. (Rust also uses snake case for the names of variable bindings; we just haven't used any variable bindings long enough to need underscores yet.) Here's a program containing an example function definition:

fn main() {
    println!("Hello, world!");

    another_function();
}

fn another_function() {
    println!("Another function.");
}

Function definitions in Rust always start with fn and have a set of parentheses after the function name. The curly braces tell the compiler where the function begins and ends.

We can call any function weve defined by entering its name followed by a pair of parentheses. Since another_function() is defined in the program, it can be called from inside the main() function. Note that we defined another_function() after the main() function in our source code; we could have defined it before as well. Rust doesnt care where you define your functions, only that they are defined somewhere.

Lets start a new project to explore functions further. Open a terminal, and navigate to the directory you're keeping your projects in. From there, use Cargo to generate a new project, as follows:

$ cargo new --bin functions
$ cd functions

Place the another_function() example in a file named src/main.rs and run it. You should see the following output:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
     Running `target/debug/functions`
Hello, world!
Another function.

The lines execute in the order they appear in the main() function. First, our “Hello, world!” message prints, and then another_function() is called and its message is printed.

Function Arguments

Functions can also take arguments. The following rewritten version of another_function() shows what arguments look like in Rust:

fn main() {
    another_function(5);
}

fn another_function(x: i32) {
    println!("The value of x is: {}", x);
}

Try running this program, and you should get this output:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
     Running `target/debug/functions`
The value of x is: 5

Since main() passed 5 to another_function(), the println macro put 5 where the pair of curly braces were in the format string.

Lets take a closer look at the signature of a function which takes a single argument:

fn NAME(PATTERN: TYPE) {

The parameter declaration in a single-argument function signature looks like the let bindings we used earlier. Just look at both together, and compare them:

let x: i32;
fn another_function(x: i32) {

The one difference is that in function signatures, we must declare the type. This is a deliberate decision in the design of Rust; requiring type annotations in function definitions means you almost never need to use them elsewhere in the code.

When you want a function to have multiple arguments, just separate them inside the function signature with commas, like this:

fn NAME(PATTERN: TYPE, PATTERN: TYPE, PATTERN: TYPE, PATTERN: TYPE...) {

And just like a let declaration with multiple patterns, a type must be applied to each pattern separately. To demonstrate, heres a full example of a function with multiple arguments:

fn main() {
    another_function(5, 6);
}

fn another_function(x: i32, y: i32) {
    println!("The value of x is: {}", x);
    println!("The value of y is: {}", y);
}

In this example, we make a function with two arguments. In this case, both are i32s, but if your function has multiple arguments, they dont have to be the same time. They just happen to be in this example. Our function then prints out the values of both of its arguments.

Lets try out this code. Replace the program currently in your function project's main.rs file with the example above, and run it as follows:


$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
     Running `target/debug/functions`
The value of x is: 5
The value of y is: 6

Since 5 is passed as the x argument and 6 is passed as the y argument, the two strings are printed with these values.

Bindings as Arguments

It's also possible to create bindings and pass them in as arguments in Rust. For example:

fn main() {
    let a = 5;
    let b = 6;

    another_function(a, b);
}

fn another_function(x: i32, y: i32) {
    println!("The value of x is: {}", x);
    println!("The value of y is: {}", y);
}

Instead of passing 5 and 6 directly, this first creates two bindings containing the values, and passes those bindings instead. When you run this, you'll find that it has the same effect as just using integers:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
     Running `target/debug/functions`
The value of x is: 5
The value of y is: 6

Note that our bindings are called a and b, yet inside the function, we refer to them by the names in the signature, x and y. Inside a function, its parameters are in scope but the names of the bindings we passed as parameters are not, so we need to use the parameter names within the function block. Bindings passed as parameters dont need to have the same names as the arguments.

Functions with Return Values

Functions can return values back to functions that call them. The signature for a function that returns a value looks like this:

fn NAME(PATTERN: TYPE, PATTERN: TYPE, PATTERN: TYPE, PATTERN: TYPE...) -> TYPE {

In Rust, we dont name return values, but we do declare their type, after the arrow (->). Heres a sample program to illustrate this concept:

fn main() {
    let x = five();

    println!("The value of x is: {}", x);
}

fn five() -> i32 {
    5
}

There are no function calls, macros, or even let statements in the five() function--just the number 5 by itself. That's a perfectly valid function in Rust. Note the function's return type, too. Try running this code, and the output should look like this:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
     Running `target/debug/functions`
The value of x is: 5

The 5 in five() is actually the function's return value, which is why the return type is i32. Lets examine this in more detail. There are two important bits. First, the line let x = five(); in main() shows that we can use the return value of a function to initialize a binding.

Because the function five() returns a 5, that line is the same as saying:

let x = 5;

The second interesting bit is the five() function itself. It requires no arguments and defines the type of the return, but the body of the function is a lonely 5 with no semicolon. So far, weve ended almost every line in our programs with a semicolon, so why not here?

The answer is that the return value of a function is the value of its final expression. To explain this, we have to go over statements and expressions.

Statements and Expressions

Expressions are bits of code that evaluate to a value. Consider a simple math operation, like this:

5 + 6

Evaluating this expression results in the value: 11. In Rust, most bits of code are expressions. For example, calling a function is an expression:

foo(5)

The value is equal to the return value of the foo() function.

Statements are instructions. While expressions compute something, statements perform some action. For example, let statements bind variables, and fn declarations are statements that begin functions.

One practical difference between an expression and a statement is that you can bind an expression, but you can't bind a statement.

For example, let is a statement, so you cant assign it to another binding, as this code tries to do:

fn main() {
    let x = (let y = 6);
}

If we were to run this program, wed get an error like this:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
src/main.rs:2:14: 2:17 error: expected identifier, found keyword `let`
src/main.rs:2     let x = (let y = 6);
                           ^~~
src/main.rs:2:18: 2:19 error: expected one of `!`, `)`, `,`, `.`, `::`, `{`, or an operator, found `y`
src/main.rs:2     let x = (let y = 6);
                               ^
Could not compile `functions`.

In the same way, we can't assign a fn declaration to a binding, either.

Expressions as Return Values

So what does the way statements and expressions work have to do with return values? Well, the block that we use to create new scopes, {}, is an expression. Lets take a closer look at {} with the following signature:

{
    STATEMENT*
    EXPRESSION
}

The * by STATEMENT indicates "zero or more," meaning we can have any number of statements inside a block, followed by an expression. Since blocks are expressions themselves, we can nest blocks inside of blocks.

And since blocks return a value, we can use them in let statements. For example:

fn main() {

    let x = 5;

    let y = {
        let z = 1;

        x + z + 5
    };

    println!("The value of y is: {}", y);
}

Here, we're using a block to give us a value for the y variable. Inside that block, we create a new variable binding, z, with a let statement and give z a value. For the final expression of the block, we do some math, ultimately binding the result to y. Since x is 5 and z is 1, the calculation is 5 + 1 + 5, and so the value of the entire block is 11. This gets substituted into our let statement for y, making that statement equivalent to:

let y = 11;

Try running the program, and you should see the following output:

   Compiling functions v0.1.0 (file:///projects/functions)
     Running `target/debug/functions`
The value of y is: 11

As expected, the output string says that y is 11.

Functions Are Expressions (Or a different heading, if this one doesn't make sense.)

We also use blocks as the body of functions, for example:

fn main() {
    let x = 5;
    
    let y = {
        x + 1
    };

    println!("The value of y is: {}", y);

    let y = plus_one(x);

    println!("The value of y is: {}", y);
}

fn plus_one(x: i32) -> i32 {
    x + 1
}

In both let statements that bind values to the y variable, we use a block to produce the value. In the first case, the block is an arbitrary scope nested within the main() function. In the second, the block is the body of the plus_one() function, which is passed x as a parameter. Running this gives:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
     Running `target/debug/functions`
The value of y is: 6
The value of y is: 6

The x variable doesn't change before the new y variable is created and bound to the return value of the plus_one() function, so both println macros tell us that y is 6.

Expression Statements

Another impoortant thing to know about expressions and statements is that adding a semicolon to the end of an expression turns it into a statement. For example, look at this modified version of our plus_one() function from earlier:

fn main() {
    let x = plus_one(5);

    println!("The value of x is: {}", x);
}

fn plus_one(x: i32) -> i32 {
    x + 1;
}

Since x + 1 is the only code in the function, it should be an expression, so that the value it evaluates to can be the function's return value. But the semicolon has turned it into a statement, so running this code would give an error, as follows:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
src/main.rs:7:1: 9:2 error: not all control paths return a value [E0269]
src/main.rs:7 fn plus_one(x: i32) -> i32 {
src/main.rs:8     x + 1;
src/main.rs:9 }
src/main.rs:7:1: 9:2 help: run `rustc --explain E0269` to see a detailed explanation
src/main.rs:8:10: 8:11 help: consider removing this semicolon:
src/main.rs:8     x + 1;
                       ^
error: aborting due to previous error
Could not compile `functions`.

The main error message, "not all control paths return a value," reveals the core of the issue with this code. Statements dont evaluate to a value, but plus_one() tries to return an i32 with only a statement in the function body. In this output, Rust gives an option to rectify this: it suggests removing the semicolon, which would fix the error.

In practice, Rust programmers dont often think about these rules at this level. On a practical level, you should remember that you usually have a semicolon at the end of most lines, but ...

<-- thinking about this more, I did think of one: calling functions. foo() is an expression, and so foo() bar() is invalid. but adding ; to make them statements is okay, foo(); bar(); Maybe i can work up an example after all, what do you think? -->

Returning Multiple Values

By default, functions can only return single values. Theres a trick, however to get them to return multiple values. Remember how we used ()s to create complex bindings in the "Creating Multiple Bindings" section on page XX?

fn main() {
    let (x, y) = (5, 6);
}

Braces used in this way form a tuple, which is a collection of elements that isn't assigned a name. Tuples are also a basic data type in Rust, and we'll cover them in detail in the "Tuples" section later in this chapter. For our purposes now, we can use tuples to return multiple values from functions, as so:

fn main() {
    let (x, y) = two_numbers();

    println!("The value of x is: {}", x);
    println!("The value of y is: {}", y);
}

fn two_numbers() -> (i32, i32) {
    (5, 6)
}

Running this will give us the values:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
     Running `target/debug/functions`
The value of x is: 5
The value of y is: 6

Let's look at this more closely. First, we're assigning the return value of two_numbers() to x and y:

fn two_numbers() -> (i32, i32) {
    (5, 6)
}

In plain English, the (i32, i32) syntax translates to, “a tuple with two i32s in it." These two types are then applied to the tuple to be returned by the function block. In this case, that tuple contains the values 5 and 6. This tuple is returned, and assigned to x and y:

let (x, y) = two_numbers();

See how all these bits fit together? We call this behavior of let destructuring, because it takes the structure of the expression that follows the = and takes it apart.

Data Types in Rust

Weve seen that every value in Rust is of a certain type, which tells Rust what kind of data is being given so it knows how to work with that data. As described in the "Type Inference and Annotation" section, you can rely on Rust's ability to infer types to figure out the type of a binding, or you can annotate it explicitly if needed. In this section, we'll look at a number of types built into the language itself. We'll look at two subsets of Rust data types: scalar and compound.

Type Inference and Annotation

Rust is a statically typed language, which means that we must know the types of all bindings at compile time. However, you may have noticed that we didnt declare a type for x or y in our previous examples.

This is because Rust can often tell the type of a binding without you having to declare it. Annotating every single binding with a type can take uneccesary time and make code noisy. To avoid this, Rust uses type inference, meaning that it attempts to infer the types of your bindings from how the binding is used. Lets look at the the first let statement you wrote again:

fn main() {
    let x = 5;
}

When we bind x to 5, the compiler determines that x should be a numeric type based on the value it is bound to. Without any other information, it sets the x variable's type to i32 (a thirty-two bit integer type) by default. Well talk more about Rusts basic types in Section 3.3.

If we were to declare the type with the variable binding, that would be called a type annotation. A let statement like that would look like this:


let PATTERN: TYPE = VALUE;

The let statement now has a colon after the PATTERN, followed by the TYPE name. Note that the colon and the TYPE go after the PATTERN, not inside the pattern itself. Given this structure, here's how you'd rewrite let x = 5 to use type annotation:


fn main() {

    let x: i32 = 5;

}

This does the same thing as let x = 5 but explicitly states that x should be of the i32 type. This is a simple case, but more complex patterns with multiple bindings can use type annotation, too. A binding with two variables would look like this:


fn main() {

    let (x, y): (i32, i32) = (5, 6);

}

In the same way as we place the VALUE and the PATTERN in corresponding positions, we also match up the position of the TYPE with the PATTERN it corresponds to.

Scalar Types

A scalar type is one that represents a single value. There are four key scalar types in Rust: integers, floating point numbers, booleans, and characters. You'll likely recognize these from other programming languages, but let's jump into how they work in Rust.

Integer Types

An integer is a number without a fractional component. We've used one integer type already in this chapter, the i32 type. This type declaration indicates that the value it's associated with should be a signed integer (hence the i) for a 32-bit system. There are a number of built-in integer types in Rust, shown in Table 3-1.

Length signed unsigned
8-bit i8 u8
16-bit i16 u16
32-bit i32 u32
64-bit i64 u64
arch isize usize

Table 3-1: Integer types in Rust. The code, for example i32, is used to define a type in a function.

Each variant can be either signed or unsigned, and has an explicit size. Signed and unsigned merely refers to whether the number can be negative or positive. An unsigned number can only be positive, while a signed number can be either positive or negative. It's like writing numbers on paper: when the sign matters, a number is shown with a plus sign or minus sign, but when it's safe to assume the number is positive, it's shown with no sign. Signed numbers are stored using twos complement representation.

Finally, the isize and usize types depend on the kind of computer your program is running on: 64-bits if you're on a 64-bit architecture, and 32-bits if youre on a 32-bit architecture.

So how do you know which type of integer to use? If you're unsure, Rust's defaults are generally good choices, and integer types default to i32: its generally the fastest, even on 64-bit systems. The primary situation in which you'd need to specify isize or usize is when indexing some sort of collection, which we'll talk about in the "Arrays" section.

Floating-Point Types

Rust also has two primitive types for floating-point numbers, which are just numbers with decimal points, as usual. Rust's floating-point types are f32 and f64, which are 32 bits and 64 bits in size, respectively. The default type is f64, as its roughly the same speed as f32, but has a larger precision. Here's an example showing floating-point numbers in action:

fn main() {
    let x = 2.0; // f64

    let y: f32 = 3.0; // f32
}

Floating-point numbers are represented according to the IEEE-754 standard. The f32 type is a single-precision float, while f64 has double-precision.

Numeric Operations

Rust supports the usual basic mathematic operations youd expect for all of these number types--addition, subtraction, multiplication, division, and modulo. This code shows how you'd use each one in a let statement:

fn main() {
    // addition
    let sum = 5 + 10;

    // subtraction
    let difference = 95.5 - 4.3;

    // multiplication
    let product = 4 * 30;

    // division
    let quotient = 56.7 / 32.2;

    // modulus
    let remainder = 43 % 5;
}

Each expression in these statements uses a mathematical operator and evaluates to a single value, which is then bound to a variable.

The Boolean Type

As in most other programming languages, a boolean type has two possible values: true and false. The boolean type in Rust is specified with bool. For example:

fn main() {
    let t = true;

    let f: bool = false; // with explict type annotation
}

The main way to consume boolean values is through conditionals like an if statement. Well cover how if statements work in Rust in the "Control Flow" section of this chapter.

The Character Type

So far weve only worked with numbers, but Rust supports letters too. Rusts char type is the language's most primitive alphabetic type, and this code shows one way to use it:

fn main() {
   let c = 'z';

   let z = '';
}

Rusts char represents a Unicode Scalar Value, which means that it can represent a lot more than just ASCII. (You can learn more about Unicode Scalar Values at http://www.unicode.org/glossary/#unicode_scalar_value) A "character" isnt really a concept in Unicode, however, so your human intutition for what a "character" is may not match up with what a char is in Rust. It also means that chars are four bytes each.

Compound Types

Compound types can group multiple values of other types into another type. Rust has two primitive compound types: tuples and arrays. You can put a compound type inside a compound type as well.

Grouping Values into Tuples

Weve seen tuples already, when binding or returning multiple values at once. A tuple is a general way of grouping together some number of other values with distinct types into one compound type. The number of values is called the arity of the tuple.

We create a tuple by writing a comma-separated list of values inside parentheses. Each position in the tuple has a distinct type, as in this example:

fn main() {
    let tup: (i32, f64, u8) = (500, 6.4, 1);
}

Note that, unlike the examples of multiple bindings, here we bind the single name tup to the entire tuple, emphasizing the fact that a tuple is considered a single compound element. We can then use pattern matching to destructure this tuple value, like this:

fn main() {
    let tup: (i32, f64, u8) = (500, 6.4, 1);

    let (x, y, z) = tup;

    println!("The value of y is: {}", y);
}

In this program, we first create a tuple, and bind it to the name tup. We then use a pattern with let to take tup and turn it into three separate bindings, x, y, and z. This is called destructuring, because it breaks the single tuple into three parts.

Finally, we print the value of x, which is 6.4.

Tuple Indexing

In addition to destructuring through pattern matching, we can also access a tuple element directly by using a period (.) followed by the index of the value we want to access. For example:

fn main() {
    let x: (i32, f64, u8) = (500, 6.4, 1);

    let five_hundred = x.0;

    let six_point_four = x.1;

    let one = x.2;
}

This program creates a tuple, x, and then makes new bindings to each element by using their index.

As with most programming languages, the first index in a tuple is 0.

Single-Element Tuples

Not everything contained within parentheses is a tuple in Rust. For example, a (5) may be a tuple, or just a 5 in parentheses. To disambiguate, use a comma for single-element tuples, as in this example:

fn main() {
    let x = (5);  

    let x = (5,); 
}

In the first let statement, because (5) has no comma, it's a simple i32 and not a tuple. In the second let example, (5,) is a tuple with only one element.

Arrays

So far, weve only represented single values in a binding. Sometimes, though, its useful to bind a name to more than one value. Data structures that contain multiple values are called collections, and arrays are the first type of Rust collection well learn about.

In Rust, arrays look like this:

fn main() {
    let a = [1, 2, 3, 4, 5];
}

The values going into an array are written as a comma separated list inside square brackets. Unlike a tuple, every element of an array must have the same type.

Type Annotation for Arrays

When you specify an arrays type, you'd do so as such:

fn main() {
    let a: [i32; 5] = [1, 2, 3, 4, 5];
}

Much like in a variable binding that uses type annotation, the array's type and length come after the pattern name and a colon. This array has 5 values, which are of the i32 type. Unlike the values themselves, the type and array length are separated by a semicolon.

####Accessing and Modifying Array Elements

An array is a single chunk of memory, allocated on the stack. We can access elements of an array using indexing, like this:

fn main() {
    let a = [1, 2, 3, 4, 5];

    let first = a[0];
    let second = a[1];
}

In this example, the first variable will bind to 1 at index [0] in the array, and second will bind to 2 at index [1] in the array. Note that these values are copied out of the array and into first and second when the let statement is called. That means if the array changes after the let statements, these bindings will not, and the two variables should retain their values. For example, imagine you have the following code:

fn main() {
    let mut a = [1, 2, 3, 4, 5];

    let first = a[0];

    a[0] = 7;

    println!("The value of first is: {}", first);
}

First, notice the use of mut in the array declaration. We had to declare array a as mut to override Rust's default immutability. The line a[0] = 7; modifies the element at index 0 in the array, changing its value to 7. This happens after first is bound to the original value at index 0, so first should still be equal 1. Running the code will show this is true:

The value of first is: 1

Since a[0] didn't change until after first was assigned a value, the println macro replaced the {} with 1, as expected.

Macros and Data Structures

Now that we've discussed data structures a little, there are a couple of relevant macro concepts we should cover: the panic! macro and Debug, which is a new way of printing data to the terminal.

Rectifying Invalid Indexes with Panic!

Rust calls the panic macro when a program tries to access elements of an array (or any other data structure) and gives an invalid index. For an example, use the functions project we created on page XX and change your src/main.rs to look like this:

fn main() {
    let a = [1, 2, 3, 4, 5];

    let invalid = a[10];

    println!("The value of invalid is: {}", invalid);
}

This program tries to access an element at index 10 in the a array. If we run it, we will get an error like this:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
     Running `target/debug/functions`
thread <main> panicked at index out of bounds: the len is 5 but the index is 10, src/main.rs:4
Process didnt exit successfully: `target/debug/functions` (exit code: 101)

The output tells us that our thread panicked, and that our program didnt exit successfully. It also gives the reason: we requested an index of 10 from an array with a length of 5.

So why did this cause Rust to panic? An array knows how many elements it holds. When we attempt to access an element using indexing, Rust will check that the index we've specified is less than the array length. If the index is greater than the length, it will panic. This is our first example of Rusts safety principles in action. In many low-level languages, this kind of check is not done, and when you provide an incorrect index, invalid memory can be accessed. Rust protects us against this kind of error. We'll discuss more of Rusts error handling in Chapter xx.

Using Debug in the println Macro

So far, weve been printing values using {} in a println macro. If we try that with an array, however, we'll get an error. Say we have the following program:

fn main() {
    let a = [1, 2, 3, 4, 5];

    println!("a is: {}", a);
}

This code tries to print the a array directly, which may seem innocuous. But running it produces the following output:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
src/main.rs:4:25: 4:26 error: the trait `core::fmt::Display` is not implemented for the type `[_; 5]` [E0277]
src/main.rs:4     println!(“a is {}”, a);
                                      ^
<std macros>:2:25: 2:56 note: in this expansion of format_args!
<std macros>:3:1: 3:54 note: in this expansion of print! (defined in <std macros>)
src/main.rs:4:5: 4:28 note: in this expansion of println! (defined in <std macros>)
src/main.rs:4:25: 4:26 help: run `rustc --explain E0277` to see a detailed explanation
src/main.rs:4:25: 4:26 note: `[_; 5]` cannot be formatted with the default formatter; try using `:?` instead if you are using a format string
src/main.rs:4:25: 4:26 note: required by `core::fmt::Display::fmt`
error: aborting due to previous error

Whew! The core of the error is this part: the trait core::fmt::Display is not implemented. We havent discussed traits yet, so this is bound to be confusing! Heres all we need to know for now: println! can do many kinds of formatting. By default, {} implements a kind of formatting known as Display: output intended for direct end-user consumption. The primitive types weve seen so far implement Display, as theres only one way youd show a 1 to a user. But with arrays, the output is less clear. Do you want commas or not? What about the []s?

More complex types in the standard library do not automatically implement Display formatting. Instead, Rust implements another kind of formatting, also intended for the programmer. This formatting type is called Debug. To ask println! to use Debug formatting, we include :? in the print string, like this:

fn main() {
    let a = [1, 2, 3, 4, 5];

    println!("a is {:?}", a);
}

If you run this, it should print the five values in the a array as desired:

$ cargo run
   Compiling functions v0.1.0 (file:///projects/functions)
     Running `target/debug/functions`
a is [1, 2, 3, 4, 5]

Youll see this repeated later, with other types. Well cover traits fully in Chapter 9.

Comments

All programmers strive to make their code easy to understand, but sometimes some extra explanation is warranted. In these cases, we leave notes in our source code that the compiler will ignore. These notes are called comments.

Heres a simple comment:

// Hello, world.

In Rust, comments must start with two slashes, and will last until the end of the line. For comments that extend beyond a single line, you'll need to include // on each line, like this:

// So were doing something complicated here, long enough that we need
// multiple lines of comments to do it! Whew! Hopefully, this comment will
// explain whats going on.

Comments can also be placed at the end of lines of code:

fn main() {
    let lucky_number = 7; // Im feeling lucky today.
}

But youll more often see them above, like so:

fn main() {
    // Im feeling lucky today.
    let lucky_number = 7;
}

Thats all there is to it. Comments are not particularly complicated.

Documentation Comments

Rust has another kind of comment: a documentation comment. These comments dont affect the way that the code works, but they do work with Rusts tools. More specifically, the rustdoc tool can read documentation comments and produce HTML documentation from them. Documentation comments use an extra slash, like this:

/// The foo function doesnt really do much.
fn foo() {

}

/// Documentation comments can use
/// multiple line comments too,
/// like we did before.
fn bar() {

}

The rustdoc tool would interpret each comment in this example as documenting the thing that follows it. The first comment would be used to document the foo() function, and the second comment would document the bar() function.

Because documentation comments have semantic meaning to rustdoc, the compiler will pay attention to the placement of your documentation comments. For example, a program containing only this:

/// What am I documenting?

Will give the following compiler error:

src/main.rs:1:1: 1:27 error: expected item after doc comment
src/main.rs:1 /// What am I documenting?
              ^~~~~~~~~~~~~~~~~~~~~~~~~~

This happens because Rust expects a document comment to be associated with whatever code comes directly after it, so it sees that a document comment alone must be a mistake.

Control Flow with if

Two roads diverged in a yellow wood,
And sorry I could not travel both
And be one traveler, long I stood
And looked down one as far as I could
To where it bent in the undergrowth;

  • Robert Frost, “The Road Not Taken”

In Rust, as in most programming languages, an if expression allows us to branch our code depending on conditions. We provide a condition, and then say, if this condition is met, then run this block of code; if the condition is not met, run a different block of code (or stop the program).

Lets make a new project to explore if. Navigate to your projects directory, and use Cargo to make a new project called branches:

$ cargo new --bin branches
$ cd branches

Write this sample program using if and save it in the branches directory:

fn main() {
    let condition = true;

    if condition {
        println!("condition was true");
    } else {
        println!("condition was false");
    }
}   

The condition variable is a boolean; here, it's set to true. All if statements start with if, which is followed by a condition. The block of code we want to execute if the condition is true goes immediately after the condition, inside curly braces. These blocks are sometimes called arms. We can also include an else statement, which gives the program a block of code to execute should condition evaluate to false.

Try running this code, and you should see output like this:

$ cargo run
   Compiling branches v0.1.0 (file:///projects/branches)
     Running `target/debug/branches`
condition was true

Before we talk about whats happening here, lets try changing the value of condition to false as follows:

    let condition = false;

Run the program again, and look at the output:

$ cargo run
   Compiling branches v0.1.0 (file:///projects/branches)
     Running `target/debug/branches`
condition was false

This time the second block of code is run, the else block is executed. This is the very basic structure of if: if the condition is true, then execute some code. If its not true, then execute some other code. When an else block is included, that "other code" is the code in the else block. You could also not use an else expression, as in this example:

fn main() {
    let condition = false;

    if condition {
        println!("condition was true");
    }
}

In this case, nothing would be printed, because there is no code after the if block.

Its also worth noting that condition here must be a bool. To see what happens if the condition isn't a bool, try running this code:

fn main() {
    let condition = 5;

    if condition {
        println!("condition was five");
    }
}

The condition variable is assigned a value of 5 this time, and Rust will complain about it:

   Compiling branches v0.1.0 (file:///projects/branches)
src/main.rs:4:8: 4:17 error: mismatched types:
 expected `bool`,
    found `_`
(expected bool,
    found integral variable) [E0308]
src/main.rs:4     if condition {
                     ^~~~~~~~~
src/main.rs:4:8: 4:17 help: run `rustc --explain E0308` to see a detailed explanation
error: aborting due to previous error
Could not compile `branches`.

The error tells us that Rust expected a bool, but got an integer. Rust will not automatically try to convert non-boolean types to a boolean here. We must be explicit.

Multiple Conditions with else if

We can set multiple coniditions by combining if and else in an else if expression. For example:

fn main() {
    let number = 5;

    if number == 3 {
        println!("condition was 3");
    } else if number == 4 {
        println!("condition was 4");
    } else if number == 5 {
        println!("condition was 5");
    } else {
        println!("condition was something else");
    }
}

This program three possible paths it can take after checking the condition, and if you try running it, you should see output like this:

$ cargo run
   Compiling branches v0.1.0 (file:///projects/branches)
     Running `target/debug/branches`
condition was 5

When this program executes, it will check each if expression in turn, and execute the first body for which the condition holds true.

Using too many else if expressions can clutter your code, so if you find yourself with more than one, you may want to look at refactoring your code. In Chapter XX, we'll talk about a powerful Rust branching construct called match for these cases.

Using if in a Binding

The last detail you need to learn about if is that its an expression. That means that we can use it on the right hand side of a let binding, for instance:

fn main() {
    let condition = true;
    let number = if condition {
        5
    } else {
        6
    };

    println!("The value of number is: {}", number);
}

The number variable will be bound to a value based on the outcome of the if expression. Lets run this to see what happens:

$ cargo run
   Compiling branches v0.1.0 (file:///projects/branches)
     Running `target/debug/branches`
The value of number is: 5

Remember, blocks of code evaluate to the last expression in them, and numbers by themselves are also expressions. In this case, the value of the whole if expression depends on which block of code executes. This means that the value in both arms of the if must be the same type; in the previous example, they were both i32 integers. But what happens if the types are mismatched, as in the following example?

fn main() {
    let condition = true;

    let number = if condition {
        5
    } else {
        "six"
    };

    println!("The value of number is: {}", number);
}

The expression in one block of the if statement, is an integer and the expresion in the other block is a string. If we try to run this, well get an error:

   Compiling branches v0.1.0 (file:///projects/branches)
src/main.rs:4:18: 8:6 error: if and else have incompatible types:
 expected `_`,
    found `&static str`
(expected integral variable,
    found &-ptr) [E0308]
src/main.rs:4     let number = if condition {
src/main.rs:5         5
src/main.rs:6     } else {
src/main.rs:7         "six"
src/main.rs:8     };
src/main.rs:4:18: 8:6 help: run `rustc --explain E0308` to see a detailed explanation
error: aborting due to previous error
Could not compile `branches`.

The if and else arms have value types that are incompatible, and Rust tells us exactly where to find the problem in our program. This cant work, because variable bindings must have a single type.

Control Flow with Loops

Its often useful to be able to execute a block of code more than one time. For this, Rust has several constructs called loops. A loop runs through the code inside it to the end and then starts immediately back at the beginning. To try out loops, lets make a new project. Navigate to your projects folder and use Cargo to make a new project:

$ cargo new --bin loops
$ cd loops

There are three kinds of loops in Rust: loop, while, and for. Lets dig in.

Repeating Code with loop

The loop keyword tells Rust to execute a block of code over and over again forever, or until we explicitly kill it.

For an example, change the src/main.rs file in your loops directory to look like this:

fn main() {
    loop {
        println!("again!");
    }
}

If we run this program, well see again! printed over and over continuously until we kill the program manually. Most terminals support a keyboard shortcut, control-c, to kill a program stuck in a continual loop. Give it a try:

$ cargo run
   Compiling loops v0.1.0 (file:///projects/loops)
     Running `target/debug/loops`
again!
again!
again!
again!
^Cagain!

That ^C there is where I hit control-c. Fortunately, Rust provides another, more reliable way to break out of a loop. We can place the break keyword within the loop to tell the program when to stop executing the loop. Try this version out of the program:

fn main() {
    loop {
        println!("once!");
        break;
    }
}

If you run this program, youll see that it only executes one time:

$ cargo run
   Compiling loops v0.1.0 (file:///projects/loops)
     Running `target/debug/loops`
once!

When a Rust program hits a break statement, it will exit the current loop. This on its own is not very useful; if we wanted to print somtheing just once, we wouldn't put it in a loop. This is where conditions come in again.

Conditional Loops With while

To make break useful, we need to give our program a condition. While the condition is true, the loop runs. When the condition ceases to be true, the break code runs, stopping the loop.

Try this example:


fn main() {
    let mut number = 3;

    loop {
        if number != 0 {
            println!("{}!", number);

            number = number - 1;
        } else {
            break;
        }
    }

    println!("LIFTOFF!!!");
}

If we run this, well get:

   Compiling loops v0.1.0 (file:///projects/loops)
     Running `target/debug/loops`
3!
2!
1!
LIFTOFF!!!

This program loops three times, counting down each time. Finally, after the loop, it prints another message, then exits.

The core of this example is in the combination of these three constructs:

    loop {
        if number != 0 {
            // do stuff
        } else {
            break;
        }

We want to loop, but only while some sort of condition is true. As soon as it isn't, we want to break out of the loop. This pattern is so common that Rust has a more efficient language construct for it, called a while loop. Here's the same example, but using while instead:

fn main() {
    let mut number = 3;

    while number != 0  {
        println!("{}!", number);

        number = number - 1;
    }

    println!("LIFTOFF!!!");
}

This gets rid of a lot of nesting, and it's more clear. While a condition holds, run this code.

Looping Though a Collection with for

We can use this while construct to loop over the elements of a collection, like an array. For example:

fn main() {
    let a = [1, 2, 3, 4, 5];
    let mut index = 0;

    while index < 5 {
        println!("the value is is: {}", a[index]);

        index = index + 1;
    }
}

Here, we're counting up through the elements in the array. We start at index 0, then loop until we hit the final index of our array (that is, when index < 5 is no longer true). Running this will print out every element of the array:

$ cargo run
   Compiling loops v0.1.0 (file:///projects/loops)
     Running `target/debug/loops`
the value is: 1
the value is: 2
the value is: 3
the value is: 4
the value is: 5

All five array values appear in the terminal, as expected. Even though index will reach a value of 6 at some point, the loop stops executing before trying to fetch a sixth value from the array.

This approach is error-prone, though; we could trigger a panic! by getting the index length incorrect. It's also slow, as the compiler needs to perform the conditional check on every element on every iteration through the loop.

As a more efficient alternative, we can use a for loop. A for loop looks something like this:

fn main() {
    let a = [1, 2, 3, 4, 5];

    let mut index = 0;

    for element in a.iter() {
        println!("the value is: {}", element);
    }
}

** NOTE: see https://github.com/rust-lang/rust/issues/25725#issuecomment-166365658, we may want to change this **

If we run this, we'll see the same output as the previous example.

** I'm going to leave it at this for now until we decide how we want to do it**