blob: fecf49e55fa272be5db6ecc4e0762e906efe7192 [file] [log] [blame] [view] [edit]
## Functions
Functions are prevalent in Rust code. Youve already seen one of the most
important functions in the language: the `main` function, which is the entry
point of many programs. Youve also seen the `fn` keyword, which allows you to
declare new functions.
Rust code uses _snake case_ as the conventional style for function and variable
names, in which all letters are lowercase and underscores separate words. Heres
a program that contains an example function definition:
<span class="filename">Filename: src/main.rs</span>
```rust
{{#rustdoc_include ../listings/ch03-common-programming-concepts/no-listing-16-functions/src/main.rs}}
```
We define a function in Rust by entering `fn` followed by a function name and a
set of parentheses. The curly brackets tell the compiler where the function body
begins and ends.
We can call any function weve defined by entering its name followed by a set of
parentheses. Because `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 the source code; we could have defined it before
as well. Rust doesnt care where you define your functions, only that theyre
defined somewhere in a scope that can be seen by the caller.
Lets start a new binary project named _functions_ to explore functions further.
Place the `another_function` example in _src/main.rs_ and run it. You should see
the following output:
```console
{{#include ../listings/ch03-common-programming-concepts/no-listing-16-functions/output.txt}}
```
The lines execute in the order in which they appear in the `main` function.
First the Hello, world!” message prints, and then `another_function` is called
and its message is printed.
### Parameters
We can define functions to have _parameters_, which are special variables that
are part of a functions signature. When a function has parameters, you can
provide it with concrete values for those parameters. Technically, the concrete
values are called _arguments_, but in casual conversation, people tend to use
the words _parameter_ and _argument_ interchangeably for either the variables in
a functions definition or the concrete values passed in when you call a
function.
In this version of `another_function` we add a parameter:
<span class="filename">Filename: src/main.rs</span>
```rust
{{#rustdoc_include ../listings/ch03-common-programming-concepts/no-listing-17-functions-with-parameters/src/main.rs}}
```
Try running this program; you should get the following output:
```console
{{#include ../listings/ch03-common-programming-concepts/no-listing-17-functions-with-parameters/output.txt}}
```
The declaration of `another_function` has one parameter named `x`. The type of
`x` is specified as `i32`. When we pass `5` in to `another_function`, the
`println!` macro puts `5` where the pair of curly brackets containing `x` was in
the format string.
In function signatures, you _must_ declare the type of each parameter. This is a
deliberate decision in Rusts design: requiring type annotations in function
definitions means the compiler almost never needs you to use them elsewhere in
the code to figure out what type you mean. The compiler is also able to give
more helpful error messages if it knows what types the function expects.
When defining multiple parameters, separate the parameter declarations with
commas, like this:
<span class="filename">Filename: src/main.rs</span>
```rust
{{#rustdoc_include ../listings/ch03-common-programming-concepts/no-listing-18-functions-with-multiple-parameters/src/main.rs}}
```
This example creates a function named `print_labeled_measurement` with two
parameters. The first parameter is named `value` and is an `i32`. The second is
named `unit_label` and is type `char`. The function then prints text containing
both the `value` and the `unit_label`.
Lets try running this code. Replace the program currently in your _functions_
projects _src/main.rs_ file with the preceding example and run it using
`cargo
run`:
```console
{{#include ../listings/ch03-common-programming-concepts/no-listing-18-functions-with-multiple-parameters/output.txt}}
```
Because we called the function with `5` as the value for `value` and `'h'` as
the value for `unit_label`, the program output contains those values.
### Statements and Expressions
Function bodies are made up of a series of statements optionally ending in an
expression. So far, the functions weve covered havent included an ending
expression, but you have seen an expression as part of a statement. Because Rust
is an expression-based language, this is an important distinction to understand.
Other languages dont have the same distinctions, so lets look at what
statements and expressions are and how their differences affect the bodies of
functions.
* **Statements** are instructions that perform some action and do not return a
value.
* **Expressions** evaluate to a resultant value. Lets look at some examples.
Weve actually already used statements and expressions. Creating a variable and
assigning a value to it with the `let` keyword is a statement. In Listing 3-1,
`let y = 6;` is a statement.
<Listing number="3-1" file-name="src/main.rs" caption="A `main` function declaration containing one statement">
```rust
{{#rustdoc_include ../listings/ch03-common-programming-concepts/listing-03-01/src/main.rs}}
```
</Listing>
Function definitions are also statements; the entire preceding example is a
statement in itself. (As we will see below, _calling_ a function is not a
statement.)
Statements do not return values. Therefore, you cant assign a `let` statement
to another variable, as the following code tries to do; youll get an error:
<span class="filename">Filename: src/main.rs</span>
```rust,ignore,does_not_compile
{{#rustdoc_include ../listings/ch03-common-programming-concepts/no-listing-19-statements-vs-expressions/src/main.rs}}
```
When you run this program, the error youll get looks like this:
```console
{{#include ../listings/ch03-common-programming-concepts/no-listing-19-statements-vs-expressions/output.txt}}
```
The `let y = 6` statement does not return a value, so there isnt anything for
`x` to bind to. This is different from what happens in other languages, such as
C and Ruby, where the assignment returns the value of the assignment. In those
languages, you can write `x = y = 6` and have both `x` and `y` have the value
`6`; that is not the case in Rust.
Expressions evaluate to a value and make up most of the rest of the code that
youll write in Rust. Consider a math operation, such as `5 + 6`, which is an
expression that evaluates to the value `11`. Expressions can be part of
statements: in Listing 3-1, the `6` in the statement `let y = 6;` is an
expression that evaluates to the value `6`. Calling a function is an expression.
Calling a macro is an expression. A new scope block created with curly brackets
is an expression, for example:
<span class="filename">Filename: src/main.rs</span>
```rust
{{#rustdoc_include ../listings/ch03-common-programming-concepts/no-listing-20-blocks-are-expressions/src/main.rs}}
```
This expression:
```rust,ignore
{
let x = 3;
x + 1
}
```
is a block that, in this case, evaluates to `4`. That value gets bound to `y` as
part of the `let` statement. Note that the `x + 1` line doesnt have a semicolon
at the end, which is unlike most of the lines youve seen so far. Expressions do
not include ending semicolons. If you add a semicolon to the end of an
expression, you turn it into a statement, and it will then not return a value.
Keep this in mind as you explore function return values and expressions next.
### Functions with Return Values
Functions can return values to the code that calls them. We dont name return
values, but we must declare their type after an arrow (`->`). In Rust, the
return value of the function is synonymous with the value of the final
expression in the block of the body of a function. You can return early from a
function by using the `return` keyword and specifying a value, but most
functions return the last expression implicitly. Heres an example of a function
that returns a value:
<span class="filename">Filename: src/main.rs</span>
```rust
{{#rustdoc_include ../listings/ch03-common-programming-concepts/no-listing-21-function-return-values/src/main.rs}}
```
There are no function calls, macros, or even `let` statements in the `five`
functionjust the number `5` by itself. Thats a perfectly valid function in
Rust. Note that the functions return type is specified too, as `-> i32`. Try
running this code; the output should look like this:
```console
{{#include ../listings/ch03-common-programming-concepts/no-listing-21-function-return-values/output.txt}}
```
The `5` in `five` is the functions 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();` shows that were using the return value of a
function to initialize a variable. Because the function `five` returns a `5`,
that line is the same as the following:
```rust
let x = 5;
```
Second, the `five` function has no parameters and defines the type of the return
value, but the body of the function is a lonely `5` with no semicolon because
its an expression whose value we want to return.
Lets look at another example:
<span class="filename">Filename: src/main.rs</span>
```rust
{{#rustdoc_include ../listings/ch03-common-programming-concepts/no-listing-22-function-parameter-and-return/src/main.rs}}
```
Running this code will print `The value of x is: 6`. But if we place a semicolon
at the end of the line containing `x + 1`, changing it from an expression to a
statement, well get an error:
<span class="filename">Filename: src/main.rs</span>
```rust,ignore,does_not_compile
{{#rustdoc_include ../listings/ch03-common-programming-concepts/no-listing-23-statements-dont-return-values/src/main.rs}}
```
Compiling this code produces an error, as follows:
```console
{{#include ../listings/ch03-common-programming-concepts/no-listing-23-statements-dont-return-values/output.txt}}
```
The main error message, `mismatched types`, reveals the core issue with this
code. The definition of the function `plus_one` says that it will return an
`i32`, but statements dont evaluate to a value, which is expressed by `()`, the
unit type. Therefore, nothing is returned, which contradicts the function
definition and results in an error. In this output, Rust provides a message to
possibly help rectify this issue: it suggests removing the semicolon, which
would fix the error.