3.17.2 模块 (Module)
1. 使用模块对代码分组
使用模块方便对代码进行分组,以提高可读性和重用性,例如如下代码:
fn main() { println!("Produce something!"); }
将其中的打印代码放入到一个模块中,变成如下:
mod factory { // 创建一个模块 pub fn produce() { // 将打印函数放在模块中 println!("Produce something!"); } } fn main() { factory::produce(); // 调用模块的函数 }
2. 定义模块控制作用域和私有性
使用模块可以控制作用域和私有性,示例如下:
mod factory { pub struct PubStruct { // 该结构体被定义为公有,外部可以使用 pub i: u32, // 该字段被定义为公有,外部可以使用 } struct PrivateStruct { // 该结构体定义为私有,外部无法使用 i: u32, } // 该函数被定义为公有,外部可以使用 pub fn function1() { let p1 = PubStruct { i: 3u32 }; println!("p1 = {:?}", p1.i); let p2 = PrivateStruct { i: 3u32 }; println!("p2 = {:?}", p2.i); function2(); println!("This is a public function!"); } // 该函数被定义为私有,外部无法使用 fn function2() { let p1 = PubStruct { i: 3u32 }; println!("p1 = {:?}", p1.i); let p2 = PrivateStruct { i: 3u32 }; println!("p2 = {:?}", p2.i); println!("This is a private function!"); } } fn main() { let _p1 = factory::PubStruct { i: 3u32 }; // let _p2 = factory::PrivateStruct { i: 3u32 }; // 打开将编译出错,只能访问公有的类型 factory::function1(); // factory::function2(); // 打开编译将出错,只能访问公有的函数 }
模块中的项默认是私有的,要在模块外部使用模块内的一个项,则必须将该项前面加上pub关键字。
3. 绝对路径和相对路径
使用模块中的项时,需要通过对应的路径才能使用这个项。路径有两种形式:
- 绝对路径(absolute path):以 crate root开头的全路径;对于外部 crate 的代码,是以 crate 名开头的绝对路径,对于对于当前 crate 的代码,则以字面值 crate 开头。
- 相对路径(relative path):从当前模块开始,以 self、super 或当前模块的标识符开头。
mod parent { pub struct A(pub u32); pub mod factory { pub fn produce() { let _a1 = super::A(1u32); // 使用相对路径 let _a2 = crate::parent::A(2u32); // 使用绝对路径 println!("Produce something!"); } } } fn main() { crate::parent::factory::produce(); // 使用绝对路径调用模块factory的函数 self::parent::factory::produce(); // 使用相对路径调用模块factory的函数 parent::factory::produce(); // 使用相对路径调用模块factory的函数 }
4. 使用use关键字引入作用域
在外部使用模块中的每个项都带上路径会显得比较重复,可以使用use关键字引入路径,示例如下:
mod parent { pub struct A(pub u32); pub mod factory { pub fn produce() { let _a1 = super::A(1u32); // 使用相对路径 let _a2 = crate::parent::A(2u32); // 使用绝对路径 println!("Produce something!"); } } } mod ss { pub struct B(pub u32); pub fn print() { println!("Hello, world!"); } } fn main() { use parent::factory::produce; // 引入produce的路径,可以直接使用produce produce(); // 直接使用 use ss::*; // 引入路径,可以使用ss中的所有公有的项 let _b = B(8u32); print(); }
还可以使用as关键字为引入的项提供新的名字,示例如下:
pub mod factory { pub fn produce() { println!("Produce something!"); } } fn main() { use factory::produce as new_produce; // 通过as将produce命名为新的名字 new_produce(); // 用新名字使用factory::produce函数 }
5. 将模块拆成多个文件
我们将上面第4点中的第二个例子拆成多个文件,步骤如下:
- 创建一个factory.rs,其内容为mod factory中的内容:
#![allow(unused)] fn main() { // src/factory.rs pub fn produce() { println!("Produce something!"); } }
- 在main.rs中导出mod,如下:
// src/main.rs mod factory; // 导出factory module,module名字和文件名字同名 fn main() { use factory::produce as new_produce; // 通过as将produce命名为新的名字 new_produce(); // 用新名字使用factory::produce函数 }
整个工程的目录结构如下: