rust/src/doc/trpl/crates-and-modules.md

534 lines
16 KiB
Markdown
Raw Normal View History

2015-01-16 14:30:27 -06:00
% Crates and Modules
2014-09-25 23:20:25 -05:00
When a project starts getting large, its considered good software
2014-09-25 23:20:25 -05:00
engineering practice to split it up into a bunch of smaller pieces, and then
fit them together. Its also important to have a well-defined interface, so
2014-09-25 23:20:25 -05:00
that some of your functionality is private, and some is public. To facilitate
these kinds of things, Rust has a module system.
# Basic terminology: Crates and Modules
Rust has two distinct terms that relate to the module system: crate and
module. A crate is synonymous with a library or package in other
languages. Hence “Cargo” as the name of Rusts package management tool: you
2014-09-25 23:20:25 -05:00
ship your crates to others with Cargo. Crates can produce an executable or a
library, depending on the project.
2014-09-25 23:20:25 -05:00
Each crate has an implicit *root module* that contains the code for that crate.
2014-09-25 23:20:25 -05:00
You can then define a tree of sub-modules under that root module. Modules allow
you to partition your code within the crate itself.
As an example, lets make a *phrases* crate, which will give us various phrases
in different languages. To keep things simple, well stick to greetings and
farewells as two kinds of phrases, and use English and Japanese (日本語) as
two languages for those phrases to be in. Well use this module layout:
2014-09-25 23:20:25 -05:00
```text
2015-03-13 22:55:01 -05:00
+-----------+
+---| greetings |
| +-----------+
+---------+ |
+---| english |---+
| +---------+ | +-----------+
| +---| farewells |
+---------+ | +-----------+
| phrases |---+
2015-03-13 22:55:01 -05:00
+---------+ | +-----------+
| +---| greetings |
| +----------+ | +-----------+
+---| japanese |--+
+----------+ |
| +-----------+
+---| farewells |
+-----------+
2014-09-25 23:20:25 -05:00
```
In this example, `phrases` is the name of our crate. All of the rest are
modules. You can see that they form a tree, branching out from the crate
*root*, which is the root of the tree: `phrases` itself.
2014-09-25 23:20:25 -05:00
Now that we have a plan, lets define these modules in code. To start,
2014-09-25 23:20:25 -05:00
generate a new crate with Cargo:
```bash
$ cargo new phrases
$ cd phrases
```
If you remember, this generates a simple project for us:
```bash
$ tree .
.
├── Cargo.toml
└── src
└── lib.rs
1 directory, 2 files
```
`src/lib.rs` is our crate root, corresponding to the `phrases` in our diagram
above.
# Defining Modules
To define each of our modules, we use the `mod` keyword. Lets make our
2014-09-25 23:20:25 -05:00
`src/lib.rs` look like this:
```
mod english {
mod greetings {
}
mod farewells {
}
}
mod japanese {
mod greetings {
}
mod farewells {
}
}
```
After the `mod` keyword, you give the name of the module. Module names follow
the conventions for other Rust identifiers: `lower_snake_case`. The contents of
each module are within curly braces (`{}`).
Within a given `mod`, you can declare sub-`mod`s. We can refer to sub-modules
with double-colon (`::`) notation: our four nested modules are
`english::greetings`, `english::farewells`, `japanese::greetings`, and
`japanese::farewells`. Because these sub-modules are namespaced under their
parent module, the names dont conflict: `english::greetings` and
2014-09-25 23:20:25 -05:00
`japanese::greetings` are distinct, even though their names are both
`greetings`.
Because this crate does not have a `main()` function, and is called `lib.rs`,
Cargo will build this crate as a library:
```bash
$ cargo build
Compiling phrases v0.0.1 (file:///home/you/projects/phrases)
$ ls target/debug
build deps examples libphrases-a7448e02a0468eaa.rlib native
2014-09-25 23:20:25 -05:00
```
`libphrase-hash.rlib` is the compiled crate. Before we see how to use this
crate from another crate, lets break it up into multiple files.
2014-09-25 23:20:25 -05:00
# Multiple file crates
If each crate were just one file, these files would get very large. Its often
2014-09-25 23:20:25 -05:00
easier to split up crates into multiple files, and Rust supports this in two
ways.
Instead of declaring a module like this:
```{rust,ignore}
mod english {
// contents of our module go here
}
```
We can instead declare our module like this:
```{rust,ignore}
mod english;
```
If we do that, Rust will expect to find either a `english.rs` file, or a
2015-03-13 22:55:01 -05:00
`english/mod.rs` file with the contents of our module.
2014-09-25 23:20:25 -05:00
Note that in these files, you dont need to re-declare the module: thats
2014-09-25 23:20:25 -05:00
already been done with the initial `mod` declaration.
Using these two techniques, we can break up our crate into two directories and
seven files:
```bash
$ tree .
.
├── Cargo.lock
├── Cargo.toml
├── src
│   ├── english
│   │   ├── farewells.rs
│   │   ├── greetings.rs
│   │   └── mod.rs
│   ├── japanese
│   │   ├── farewells.rs
│   │   ├── greetings.rs
│   │   └── mod.rs
│   └── lib.rs
└── target
└── debug
├── build
├── deps
├── examples
├── libphrases-a7448e02a0468eaa.rlib
└── native
2014-09-25 23:20:25 -05:00
```
`src/lib.rs` is our crate root, and looks like this:
```{rust,ignore}
mod english;
mod japanese;
```
These two declarations tell Rust to look for either `src/english.rs` and
`src/japanese.rs`, or `src/english/mod.rs` and `src/japanese/mod.rs`, depending
on our preference. In this case, because our modules have sub-modules, weve
2014-09-25 23:20:25 -05:00
chosen the second. Both `src/english/mod.rs` and `src/japanese/mod.rs` look
like this:
```{rust,ignore}
mod greetings;
mod farewells;
```
Again, these declarations tell Rust to look for either
`src/english/greetings.rs` and `src/japanese/greetings.rs` or
`src/english/farewells/mod.rs` and `src/japanese/farewells/mod.rs`. Because
these sub-modules dont have their own sub-modules, weve chosen to make them
2014-09-25 23:20:25 -05:00
`src/english/greetings.rs` and `src/japanese/farewells.rs`. Whew!
The contents of `src/english/greetings.rs` and `src/japanese/farewells.rs` are
both empty at the moment. Lets add some functions.
2014-09-25 23:20:25 -05:00
Put this in `src/english/greetings.rs`:
```rust
fn hello() -> String {
"Hello!".to_string()
}
2014-09-25 23:20:25 -05:00
```
Put this in `src/english/farewells.rs`:
```rust
fn goodbye() -> String {
"Goodbye.".to_string()
}
2014-09-25 23:20:25 -05:00
```
Put this in `src/japanese/greetings.rs`:
```rust
fn hello() -> String {
"こんにちは".to_string()
}
2014-09-25 23:20:25 -05:00
```
Of course, you can copy and paste this from this web page, or just type
something else. Its not important that you actually put konnichiwa to learn
2014-09-25 23:20:25 -05:00
about the module system.
Put this in `src/japanese/farewells.rs`:
```rust
fn goodbye() -> String {
"さようなら".to_string()
}
2014-09-25 23:20:25 -05:00
```
(This is Sayōnara, if youre curious.)
2014-09-25 23:20:25 -05:00
Now that we have some functionality in our crate, lets try to use it from
2014-09-25 23:20:25 -05:00
another crate.
# Importing External Crates
We have a library crate. Lets make an executable crate that imports and uses
2014-09-25 23:20:25 -05:00
our library.
Make a `src/main.rs` and put this in it (it wont quite compile yet):
2014-09-25 23:20:25 -05:00
```rust,ignore
extern crate phrases;
fn main() {
println!("Hello in English: {}", phrases::english::greetings::hello());
println!("Goodbye in English: {}", phrases::english::farewells::goodbye());
println!("Hello in Japanese: {}", phrases::japanese::greetings::hello());
println!("Goodbye in Japanese: {}", phrases::japanese::farewells::goodbye());
}
```
The `extern crate` declaration tells Rust that we need to compile and link to
the `phrases` crate. We can then use `phrases` modules in this one. As we
2014-09-25 23:20:25 -05:00
mentioned earlier, you can use double colons to refer to sub-modules and the
functions inside of them.
Also, Cargo assumes that `src/main.rs` is the crate root of a binary crate,
rather than a library crate. Our package now has two crates: `src/lib.rs` and
2014-09-25 23:20:25 -05:00
`src/main.rs`. This pattern is quite common for executable crates: most
functionality is in a library crate, and the executable crate uses that
library. This way, other programs can also use the library crate, and its also
2014-09-25 23:20:25 -05:00
a nice separation of concerns.
This doesnt quite work yet, though. We get four errors that look similar to
2014-09-25 23:20:25 -05:00
this:
```bash
$ cargo build
Compiling phrases v0.0.1 (file:///home/you/projects/phrases)
src/main.rs:4:38: 4:72 error: function `hello` is private
src/main.rs:4 println!("Hello in English: {}", phrases::english::greetings::hello());
^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
2014-09-25 23:20:25 -05:00
note: in expansion of format_args!
<std macros>:2:25: 2:58 note: expansion site
<std macros>:1:1: 2:62 note: in expansion of print!
<std macros>:3:1: 3:54 note: expansion site
<std macros>:1:1: 3:58 note: in expansion of println!
phrases/src/main.rs:4:5: 4:76 note: expansion site
2014-09-25 23:20:25 -05:00
```
By default, everything is private in Rust. Lets talk about this in some more
2014-09-25 23:20:25 -05:00
depth.
# Exporting a Public Interface
Rust allows you to precisely control which aspects of your interface are
public, and so private is the default. To make things public, you use the `pub`
keyword. Lets focus on the `english` module first, so lets reduce our `src/main.rs`
2014-09-25 23:20:25 -05:00
to just this:
```{rust,ignore}
extern crate phrases;
fn main() {
println!("Hello in English: {}", phrases::english::greetings::hello());
println!("Goodbye in English: {}", phrases::english::farewells::goodbye());
}
```
In our `src/lib.rs`, lets add `pub` to the `english` module declaration:
2014-09-25 23:20:25 -05:00
```{rust,ignore}
pub mod english;
mod japanese;
```
And in our `src/english/mod.rs`, lets make both `pub`:
2014-09-25 23:20:25 -05:00
```{rust,ignore}
pub mod greetings;
pub mod farewells;
```
In our `src/english/greetings.rs`, lets add `pub` to our `fn` declaration:
2014-09-25 23:20:25 -05:00
```{rust,ignore}
pub fn hello() -> String {
"Hello!".to_string()
}
```
And also in `src/english/farewells.rs`:
```{rust,ignore}
pub fn goodbye() -> String {
"Goodbye.".to_string()
}
```
Now, our crate compiles, albeit with warnings about not using the `japanese`
functions:
```bash
$ cargo run
Compiling phrases v0.0.1 (file:///home/you/projects/phrases)
src/japanese/greetings.rs:1:1: 3:2 warning: function is never used: `hello`, #[warn(dead_code)] on by default
src/japanese/greetings.rs:1 fn hello() -> String {
src/japanese/greetings.rs:2 "こんにちは".to_string()
src/japanese/greetings.rs:3 }
src/japanese/farewells.rs:1:1: 3:2 warning: function is never used: `goodbye`, #[warn(dead_code)] on by default
src/japanese/farewells.rs:1 fn goodbye() -> String {
src/japanese/farewells.rs:2 "さようなら".to_string()
src/japanese/farewells.rs:3 }
Running `target/debug/phrases`
2014-09-25 23:20:25 -05:00
Hello in English: Hello!
Goodbye in English: Goodbye.
```
Now that our functions are public, we can use them. Great! However, typing out
`phrases::english::greetings::hello()` is very long and repetitive. Rust has
another keyword for importing names into the current scope, so that you can
refer to them with shorter names. Lets talk about `use`.
2014-09-25 23:20:25 -05:00
# Importing Modules with `use`
Rust has a `use` keyword, which allows us to import names into our local scope.
Lets change our `src/main.rs` to look like this:
2014-09-25 23:20:25 -05:00
```{rust,ignore}
extern crate phrases;
use phrases::english::greetings;
use phrases::english::farewells;
fn main() {
println!("Hello in English: {}", greetings::hello());
println!("Goodbye in English: {}", farewells::goodbye());
}
```
The two `use` lines import each module into the local scope, so we can refer to
the functions by a much shorter name. By convention, when importing functions, its
2014-09-25 23:20:25 -05:00
considered best practice to import the module, rather than the function directly. In
other words, you _can_ do this:
```{rust,ignore}
extern crate phrases;
use phrases::english::greetings::hello;
use phrases::english::farewells::goodbye;
fn main() {
println!("Hello in English: {}", hello());
println!("Goodbye in English: {}", goodbye());
}
```
2015-03-01 22:45:53 -06:00
But it is not idiomatic. This is significantly more likely to introduce a
naming conflict. In our short program, its not a big deal, but as it grows, it
2014-09-25 23:20:25 -05:00
becomes a problem. If we have conflicting names, Rust will give a compilation
error. For example, if we made the `japanese` functions public, and tried to do
this:
```{rust,ignore}
extern crate phrases;
use phrases::english::greetings::hello;
use phrases::japanese::greetings::hello;
fn main() {
println!("Hello in English: {}", hello());
println!("Hello in Japanese: {}", hello());
}
```
Rust will give us a compile-time error:
```text
2014-09-25 23:20:25 -05:00
Compiling phrases v0.0.1 (file:///home/you/projects/phrases)
src/main.rs:4:5: 4:40 error: a value named `hello` has already been imported in this module [E0252]
src/main.rs:4 use phrases::japanese::greetings::hello;
^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
2014-09-25 23:20:25 -05:00
error: aborting due to previous error
Could not compile `phrases`.
```
If were importing multiple names from the same module, we dont have to type it out
2015-03-13 22:55:01 -05:00
twice. Instead of this:
2014-09-25 23:20:25 -05:00
```{rust,ignore}
use phrases::english::greetings;
use phrases::english::farewells;
```
2015-03-13 22:55:01 -05:00
We can use this shortcut:
2014-09-25 23:20:25 -05:00
```{rust,ignore}
use phrases::english::{greetings, farewells};
```
## Re-exporting with `pub use`
You dont just use `use` to shorten identifiers. You can also use it inside of your crate
2014-09-25 23:20:25 -05:00
to re-export a function inside another module. This allows you to present an external
interface that may not directly map to your internal code organization.
Lets look at an example. Modify your `src/main.rs` to read like this:
2014-09-25 23:20:25 -05:00
```{rust,ignore}
extern crate phrases;
use phrases::english::{greetings,farewells};
use phrases::japanese;
fn main() {
println!("Hello in English: {}", greetings::hello());
println!("Goodbye in English: {}", farewells::goodbye());
println!("Hello in Japanese: {}", japanese::hello());
println!("Goodbye in Japanese: {}", japanese::goodbye());
}
```
Then, modify your `src/lib.rs` to make the `japanese` mod public:
```{rust,ignore}
pub mod english;
pub mod japanese;
```
Next, make the two functions public, first in `src/japanese/greetings.rs`:
```{rust,ignore}
pub fn hello() -> String {
"こんにちは".to_string()
}
```
And then in `src/japanese/farewells.rs`:
```{rust,ignore}
pub fn goodbye() -> String {
"さようなら".to_string()
}
```
Finally, modify your `src/japanese/mod.rs` to read like this:
```{rust,ignore}
pub use self::greetings::hello;
pub use self::farewells::goodbye;
mod greetings;
mod farewells;
```
The `pub use` declaration brings the function into scope at this part of our
module hierarchy. Because weve `pub use`d this inside of our `japanese`
2014-09-25 23:20:25 -05:00
module, we now have a `phrases::japanese::hello()` function and a
`phrases::japanese::goodbye()` function, even though the code for them lives in
`phrases::japanese::greetings::hello()` and
`phrases::japanese::farewells::goodbye()`. Our internal organization doesnt
2014-09-25 23:20:25 -05:00
define our external interface.
2015-03-14 18:09:26 -05:00
Here we have a `pub use` for each function we want to bring into the
`japanese` scope. We could alternatively use the wildcard syntax to include
2015-03-14 18:09:26 -05:00
everything from `greetings` into the current scope: `pub use self::greetings::*`.
What about the `self`? Well, by default, `use` declarations are absolute paths,
starting from your crate root. `self` makes that path relative to your current
place in the hierarchy instead. Theres one more special form of `use`: you can
`use super::` to reach one level up the tree from your current location. Some
people like to think of `self` as `.` and `super` as `..`, from many shells
display for the current directory and the parent directory.
Outside of `use`, paths are relative: `foo::bar()` refers to a function inside
of `foo` relative to where we are. If thats prefixed with `::`, as in
`::foo::bar()`, it refers to a different `foo`, an absolute path from your
crate root.
2014-09-25 23:20:25 -05:00
Also, note that we `pub use`d before we declared our `mod`s. Rust requires that
`use` declarations go first.
This will build and run:
```bash
$ cargo run
2014-09-25 23:20:25 -05:00
Compiling phrases v0.0.1 (file:///home/you/projects/phrases)
Running `target/debug/phrases`
2014-09-25 23:20:25 -05:00
Hello in English: Hello!
Goodbye in English: Goodbye.
Hello in Japanese: こんにちは
Goodbye in Japanese: さようなら
```