Rust
Rust 由 Mozila 一位工程师创造,他对这个语言的期望是:安全,性能,特性广泛,以内存安全为第一原则,注重并发安全,持续提升性能,现代化语言特性,拥抱开源社区。Rust 是通用型语言,适用所有领域绝大数场景,但本质是弥补 C 的内存安全问题,因为百分之 70 的安全漏洞都是非法访问内存引起。Rust 将会为各领域的基础设施做出贡献,但也有可能会出现杀手级应用
安装 && 编译 && 运行
一般不单独安装 Rust 的编译器,而是使用 rustup 安装 Rust 相关的一整套工具链:编译器,标准库,Cargo 等
提示
可以为 rustup 添加国内源,详见清华镜像源
RUSTUP_UPDATE_ROOT=https://mirrors.tuna.tsinghua.edu.cn/rustup/rustup
RUSTUP_DIST_SERVER=https://mirrors.tuna.tsinghua.edu.cn/rustup
在 Windows 安装 Rust 需要有预备环境Microsoft C++ 生成工具,保持最小安装的组件为:MSVC C++ Build 和 Windows SDK
编写一个代码,使用rustc main.rs
进行编译生成可执行程序,最后执行生成的可执行程序
fn main() {
println!("hello, world!");
}
Cargo
Cargo 是官方构建工具,用来管理 Rust 项目,使用cargo new <name>
创建项目,构建项目使用cargo build
,构建结果在target
下,cargo run
或直接run
会编译并运行可执行文件
为 Cargo 添加国内源
添加$CARGO_HOME/config.toml
文件,将仓库默认地址替换为国内的地址,加快下载依赖的速度
[source.crates-io]
registry = "https://github.com/rust-lang/crates.io-index"
replace-with = 'rsproxy-sparse'
# 清华
[source.tuna]
registry = "https://mirrors.tuna.tsinghua.edu.cn/git/crates.io-index.git"
[source.tuna-sparse]
registry = "sparse+https://mirrors.tuna.tsinghua.edu.cn/crates.io-index/"
# 中科大
[source.ustc]
registry = "git://mirrors.ustc.edu.cn/crates.io-index"
[source.ustc-sparse]
registry = "sparse+https://mirrors.ustc.edu.cn/crates.io-index/"
# 字节
[registries.rsproxy]
index = "https://rsproxy.cn/crates.io-index"
[source.rsproxy-sparse]
registry = "sparse+https://rsproxy.cn/index/"
.
├── .git
├── .gitignore
├── Cargo.toml
└── src
└── main.rs
Cargo 常用命令
cargo add <crates>
添加依赖项到Cargo.toml
cargo update
更新依赖,可以增加-p
指定依赖cargo clean
清除编译产物cargo check
在开发过程中检查代码cargo install
安装可执行程序
Cargo.toml
使用 TOML (Tom's Obvious, Minimal Language) 格式,这是 Cargo 配置文件的格式。[package]
是一个片段(section)标题,表明下面的语句用来配置一个包。[dependencies]
是项目依赖片段的开始
[package]
name = "hello_world" # 定义项目名字
version = "0.1.0" # 定义项目版本
edition = "2021" # 声明使用的 Rust 大版本,2015 2018 2021
[dependencies]
提示
可以在这里看到更多关于清单的描述
Crate
在 Rust 中,代码包被称为 crate,在Cargo.toml
中主要通过以下方式描述项目的依赖:
- 基于官方仓库,通过版本描述
- 基于项目的 Git 仓库地址,通过 URL 描述
- 基于本地项目的路径,通过类 Unix 的模式描述
[dependencies]
rand = "0.3"
hammer = { version = "0.5.0"}
color = { git = "https://github.com/bjz/color-rs" }
geometry = { path = "crates/geometry" }
提示
如果想要知道依赖的用法,可以使用cargo doc --open
,如果不想生成依赖就加上--no-deps
参数
Cargo.lock
Cargo.lock 文件在 Rust 项目中有以下几个主要作用:
- 锁定 crate 依赖的版本:Cargo.lock 记录了项目第一次执行 cargo check/build/test 时将哪个具体版本的依赖包拉取下来,并锁定依赖包不会随 cargo update 变更版本
- 重复构建:当重复运行 cargo build 时,Cargo 会用 Cargo.lock 中记录的版本来建立包依赖关系,不会重新下载版本。保证重复构建结果一致
- CI/CD 环境:通过提交 Cargo.lock 到版本控制中可以确保其他开发者的 CI 环境在构建项目时使用完全一致的包依赖版本
- 孤立环境构建:在没有联网的环境下,通过 Cargo.lock 记录的本地依赖版本信息可以完成项目构建
- 依赖升级:cargo update 会更新 Cargo.lock 中的依赖版本,方便跟踪依赖是否已经升级
Rustfmt
Rustfmt 是格式化工具,可以为项目添加一个rustfmt.toml
进行配置
max_width = 100 // 设置最大行宽为 100 个字符
tab_spaces = 2 // 设置缩进宽度为 2 个空格
然后执行cargo fmt
进行格式化
Clippy
Clippy 是 Rust 的 lint 工具
对整个项目进行静态检查
cargo clippy
自动修复 Clippy 指出的问题
cargo fix
Release
如果想发布一个 crate 到 crate.io,那么 cargo-release 是一个很好的工具
# 执行一次发布演习
cargo release [major|minor|patch] --registry crates-io
# 执行一次真实的发布
cargo release [major|minor|patch] --registry crates-io --execute
在这之前,完全可以配合git-cliff
做到发布自动生成日志,首先要在项目根目录中创建release.toml
写下预发布钩子
pre-release-hook = [
"git",
"cliff",
"-o",
"CHANGELOG.md",
"--tag",
"{{version}}",
]
Workspace
工作空间用于将代码拆分多个代码包,Cargo.toml
有所不同,以workspace
区域开始,同时指定二进制包的位置来添加成员
[workspace]
members=[
"main"
]
此时整个目录如下:
├── Cargo.toml
├── main
│ ├── Cargo.toml
│ └── src
│ └── main.rs
└── target
cargo build
会将所有成员的产物放在根目录的target
中
添加一个库包cargo new add-one --lib
,此时目录如下:
├── Cargo.toml
├── add-one
│ ├── Cargo.toml
│ └── src
│ └── lib.rs
└── main
├── Cargo.toml
└── src
└── main.rs
Cargo 不会自动引入依赖,所以需要主动引入包之间的依赖关系
[package]
name = "main"
version = "0.1.0"
edition = "2021"
[workspace]
members=[
"main",
"add-one"
]
[dependencies]
add-one = { path = "../add-one" }
为了在根目录运行二进制包,需要使用cargo run -p main
来指定
整个工作空间只会在根目录下有Cargo.lock
文件,即使在每个包中添加相同的依赖包,Cargo 也不会重复下载,保证相同的包都是一个版本,节约了磁盘空间,确保了包之间彼此兼容
除此之外,还有不包含的[package]
的工作目录,被称为虚拟清单(virtual manifest)。对于没有主package
的场景且希望将所有的package
组织在一起时非常适合,虚拟清单是不能够添加dependencies
的
Build Scripts
构建脚本本质上是一个名为build.rs
的 Rust 程序,它在编译主项目之前运行,主要用于:
- 生成代码 - 可以动态生成 Rust 代码,这在处理复杂的宏或者从其他格式(如 JSON 或 YAML)生成代码时很有用
- 编译本地代码 - 如果项目依赖于 C 或 C++ 代码,可以使用构建脚本来编译这些代码
- 配置条件编译 - 可以基于环境变量或系统特性设置条件编译标志
- 查找系统库 - 可以检测系统中是否存在某些库,并相应地配置项目
- 设置链接器参数 - 可以指定如何链接外部库
这是一个使用 ffi 调用 C 库的例子:
// build.rs
fn main() {
// 默认情况下,build.rs 不会每次都重新编译执行,但可以使用 stdout 来和 Cargo 进行通信告知何时重新运行构建脚本
println!("cargo:rerun-if-changed=src/example.c");
cc::Build::new().file("src/example.c").compile("example");
}
// main.rs
// 外部函数接口(FFI)声明,“C”指定使用了 C 语言的 ABI
// #[link(name = "example")] // 用于告诉 Rust 编译器需要链接哪个外部库,由于使用了 cc,所以自动处理了链接,因此可以忽略
extern "C" {
fn c_add(a: i32, b: i32) -> i32;
}
fn main() {
unsafe { println!("{}", c_add(1, 2)) };
}
# 只会在构建时使用的依赖
[build-dependencies]
cc = "1.0.66"