类似es6的模块化,Rust通过package、create、module来实现代码的模块化管理
Rust的代码组织包括:哪些细节可以暴露,哪些细节是私有的,作用域内哪些名称有效等等。
而这些功能被统称为模块系统,模块系统被分为(由上到下层层包含):
create的类型:
其中,关于Create,还有个概念——Create Root:
一个Package:
我们使用cargo新建一个项目
然后会提示: Created binary (application) my-project
package,这代表我们创建了一个二进制的应用程序,名叫my-project
的package
我们进入这个文件夹:
我们可以看到src/min.rs文件,这是我们程序的入口文件,但是我们在Cargo.toml中并没有看到相关的配置:
[package]
name = "my-project"
version = "0.1.0"
edition = "2021"
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
[dependencies]
这是因为cargo有一些惯例
src/main.rs是binary create的create root* create的名与package名相同如果我们还有一个这个文件:src/lib.rs,那么:
表明package包含一个library create
它是library create的create root
create的名与package名相同
Cargo将会把create root文件交给rustc(rust编译器)来构建library或者binary
一个Package可以同时包含src/main.rs和src/lib.rs
一个Package也可以有多个binary create:
将相关功能组合到一个作用域内,便于在项目间进行共享。
同时,这也能防止命名冲突,例如rand create,访问它的功能需要通过它的名字:rand
Module:
建立module:
mod front_of_house {mod hosting {fn add_to_waitlist() {}fn seat_at_table() {}}mod serving {fn take_order() {}fn serve_order() {}fn take_payment() {}}
}
src/main.rs 和 src/lib.rs 叫做create roots:
路径的作用是为了在rust的模块中找到某个条目
路径的两种形式:
路径至少由一个标识符组成,标识符之间使用::
。
举个例子(下面这段程序将报错,我们将在后面讲到如何解决):
mod front_of_house {mod hosting {fn add_to_waitlist() {}}
}
pub fn eat_at_restaurant() {crate::front_of_house::hosting::add_to_waitlist();//绝对路径front_of_house::hosting::add_to_waitlist();//相对路径
}
那么为什么会报错呢?
我们查看报错的原因:module hosting
is private,编译器告诉我们,hosting这个module是私有的。至此,为了解决这个问题,我们应该去了解一下私有边界。
为什么rust默认这些条目是私有的呢?因为rust希望能够隐藏内部的实现细节,这样就会让开发者明确知道:更改哪些内部代码的时候,不会破坏外部的代码。同时,我们可以使用pub关键字将其声明为公共的。
rust默认这些条目为私有的,我们可以使用pub
关键字来将某些条目标记为公共的。
我们将hosting
声明pub,add_to_waitlist
这个function也要声明pub
mod front_of_house {pub mod hosting {pub fn add_to_waitlist() {}}
}
pub fn eat_at_restaurant() {crate::front_of_house::hosting::add_to_waitlist();//绝对路径front_of_house::hosting::add_to_waitlist();//相对路径
}
为什么front_of_house
这个mod不需要添加pub呢?因为它们是同级的。
super:用来访问父级模块路径中的内容,类似文件系统中的..
fn serve_order() {}
mod front_of_house {fn fix_incorrect_order() {cook_order();super::serve_order();}fn cook_order() {}
}
声明一个公共的struct就是将pub放在struct前:
mod back_of_house {pub struct Breakfast {}
}
声明了一个公共的struct后:
而我们想让struct中的字段为公有的必须在前面加上pub
:
mod back_of_house {pub struct Breakfast {pub toast: String,//公有的seasonal_fruit: String, //私有的}
}
也就是说:struct的字段需要单独设置pub来变成公有
我们看一个例子:
mod back_of_house {pub struct Breakfast {pub toast: String,//公有的seasonal_fruit: String, //私有的}impl Breakfast {//一个关联函数pub fn summer(toast: &str) -> Breakfast {Breakfast {toast: String::from(toast),seasonal_fruit: String::from("peaches"),}}}
}
pub fn eat_at_restaurant() {let mut meal = back_of_house::Breakfast::summer("Rye");meal.toast = String::from("Wheat");println!("I'd like {} toast please", meal.toast);meal.seasonal_fruit = String::from("blueberries");//报错:field `seasonal_fruit` is private
}
声明一个公共的enum就是将pub放在enum前:
mod back_of_house {pub enum Appetizer {}
}
我们声明了一个公共的enum后:
mod back_of_house {pub enum Appetizer {Soup,//公共的Salad, //公共的}
}
为什么呢?因为枚举里面只有变体,只有变体是公共的这个枚举才有用。而struct中某些部分为私有的也不影响struct的使用,所以rust规定公共的struct中的字段默认为私有的。
我们可以使用use
关键字将路径导入到作用域内,而我们引入的东西也任然遵循私有性规则(公共的引入的才能用)
mod front_of_house {pub mod hosting {pub fn add_to_waitlist() {}fn some_function() {}//私有的,使用use导入后,外部依然不能调用这个函数}
}
use crate::front_of_house::hosting;
// 相当于mod hosting {}
pub fn eat_at_restaurant() {hosting::add_to_waitlist();hosting::add_to_waitlist();hosting::add_to_waitlist();
}
使用use来指定相对路径(和使用条目时的规则相同):
use front_of_house::hosting;
我们可以注意到我们调用的add_to_waitlist
是导入的hosting
mod下的,那我们可不可以直接导入function呢?
当然是可以的(不过并不推荐直接导入方法):
mod front_of_house {pub mod hosting {pub fn add_to_waitlist() {}}
}
use crate::front_of_house::hosting::add_to_waitlist;
// 相对于mod hosting {}
pub fn eat_at_restaurant() {add_to_waitlist();
}
当我们直接导入方法时,我们有可能就搞不清楚是从其他模块导入的还是在这个作用域下声明的。
所以,通常情况下,我们导入的通常为父级模块。
//...
use crate::front_of_house::hosting;
pub fn eat_at_restaurant() {hosting::add_to_waitlist();
}
不过,struct,enum,其他:指定完整路径(指定到本身)
use std::collections::HashMap;
fn main() {let mut map = HashMap::new();map.insert(1, 2);
}
但是同名的条目,我们在引入时需指定父级模块(比如下面的例子,两个类型都叫Result)
use std::fmt;
use std::io;
fn f1() -> fmt::Result {//...
}
fn f2() -> io::Result {//...
}
//...
关于上面同名的问题,还有另一种解决方法:使用as关键字
as关键字可以为引入的路径指定本地的别名
use std::fmt::Result;
use std::io::Result as IoResult;
fn f1() -> Result {//...
}
fn f2() -> IoResult {//...
}
使用 use 将路径(名称)导入到作用域内后,该名称在此作用域内是私有的,外部的模块是没办法访问use导入的模块的。
由前面pub的作用可知,类似pub fn、pub mod,我们可以使用pub use
来导入,相当于它导入了这个内容,然后又将它导出了。
(当我们使用pub use时会发现没有警告:“导入了但没有使用”,因为它同时也导出了,也被视作使用了这个导入的内容)
我们通过在Cargo.toml中的[dependencies]
添加依赖:
# ...
[dependencies]
rand = "^0.8.5"
出现:Blocking waiting for file lock on package cache
删除User/.cargo文件夹中的.package-cache
文件。重新执行cargo build
下载依赖。
很多时候我们的下载速度很慢,我们可以将下载源换到国内,在用户文件夹下的.cargo
文件夹中添加 config 文件,写入以下内容:
[source.crates-io]
registry = "https://github.com/rust-lang/crates.io-index"
replace-with = 'ustc'
[source.ustc]
registry = "git://mirrors.ustc.edu.cn/crates.io-index"
# 如果所处的环境中不允许使用 git 协议,可以把上面的地址改为
# registry = "https://mirrors.ustc.edu.cn/crates.io-index"
#[http]
#check-revoke = false
这时候cargo build就会很快了。
我们这样导入:
use rand::Rng;
另外:标准库也被当做外部包,需要导入,并且:
use std::{ascii, io};
//相当于:use std::ascii;
// use std::io;
这样的导入该如何简写呢?
use std::io;
use std::io::Chain;
可以使用self
use std::io::{self, Chain};
假如我们的src/lib.rs中的内容是这样:
mod front_of_house {pub mod hosting {pub fn add_to_waitlist() {}}
}
//...
我们可以在lib.rs同级目录下新建front_of_house.rs
,然后将模块内容写在文件中:
front_of_house.rs
pub mod hosting {pub fn add_to_waitlist() {}
}
lib.rs
mod front_of_house;
//...
如果我们想将hosting
模块的内容单独存放呢?
我们需要新建一个front_of_house文件夹,并新建hosting.rs文件
hosting.rs
pub fn add_to_waitlist() {}
front_of_house.rs
pub mod hosting;
lib.rs
mod front_of_house;
//...
原来的文件内容:
mod front_of_house {pub mod hosting {pub fn add_to_waitlist() {}}
}
随着模块逐渐变大,这项功能将能够帮助我们更好的管理代码
整理了一套《前端大厂面试宝典》,包含了HTML、CSS、JavaScript、HTTP、TCP协议、浏览器、VUE、React、数据结构和算法,一共201道面试题,并对每个问题作出了回答和解析。
有需要的小伙伴,可以点击文末卡片领取这份文档,无偿分享
部分文档展示:
文章篇幅有限,后面的内容就不一一展示了
有需要的小伙伴,可以点下方卡片免费领取