学Rust不学Cargo,等于没学Rust:workspace详解

本文涉及的产品
对象存储 OSS,20GB 3个月
对象存储 OSS,内容安全 1000次 1年
阿里云盘企业版 CDE,企业版用户数5人 500GB空间
简介: Rust 中的`Workspace`是一种组织多个 Rust crate(项目或库)的结构。使得它们可以协同工作、共享依赖关系,以及更方便地进行管理和构建。如果你是Java开发者,workspace这个概念类似Java中的`maven`父工程。子工程可以共享父工程中的很多配置项,如依赖,版本等配置。子工程可以`选择性的继承`父工程的配置。

上一篇文章我们介绍了Cargo.toml中的[features]配置块,这次我们再来看看[workspace]配置块的用法。

Rust 中的Workspace是一种组织多个 Rust crate(项目或库)的结构。使得它们可以协同工作、共享依赖关系,以及更方便地进行管理和构建。
如果你是Java开发者,workspace这个概念类似Java中的maven父工程。
子工程可以共享父工程中的很多配置项,如依赖,版本等配置。子工程可以选择性的继承父工程的配置。

一个workspace组织的项目通常如下:

│  Cargo.lock
│  Cargo.toml (Workspace configuration)
├─target/
├─bin
│  │  Cargo.toml
│  │
│  └─src
│          main.rs
│
└─osk-json-lib
    │  Cargo.toml
    │
    └─src
            lib.rs

工程根目录下Cargo.toml是配置当前workspace, 同时该目录下还有多个子包(crate),例如:bin、osk-json-lib。所有的crate共同编译到根目录下的target,且使用同一个Cargo.lock。

下面我们看一下工程根目录的Cargo.toml

[workspace]
resolver="2"
members = [
    "osk-json-lib",
    "bin"
]

[workspace.package] # 共享的package配置项
version = "1.2.3" 
authors = ["Nice Folks"] 
description = "A short description of my package" 
documentation = "https://example.com/bar"

[workspace.dependencies] # 共享的依赖项
cc = "1.0.73" 
rand = "0.8.5" 
regex = { version = "1.6.0", default-features = false, features = ["std"] }

[workspace.metadata.webcontents] 
root = "path/to/webproject" 
tool = ["npm", "run", "build"]

Workspace自己的配置,都定义在[workspace]配置项里面,包含:

  • members: 当前workspace中包含哪些crate。
  • resolver:当前workspace使用哪个版本的解析器。
  • exclude:不常用,当前workspace中排除的crate,当目录中有crate不属于这个workspace可以使用这个属性排除掉。
  • default-members:不常用,类似members。

resolver是什么?

resolver用于指定当前workspace使用的依赖解析器版本,目前有两个版本:版本1,版本2。

Rust作为一门现代语言,在语言迭代过程中会引入一些不兼容的语法。同一个只能使用一个版本的语法,因此crate在创建时就要指定采用那一版语法。目前有三个版本:Edition2015、Edition2018、Edition2021

Edition2021和以后的依赖解析器默认是版本2,之前的是版本1。

crate指定edition选项,Cargo就能识别到依赖解析器的版本。然而workspace没有这个选项,因此注明resolver="2",就表示采用的是版本2.

1. 共享的package配置项

在根工程的Cargo.toml中我们看到[workspace.package]配置项,它的作用是让子包(crate)可以共享package属性。

/osk-json-lib/Cargo.toml :

[package] 
name = "bar" 
version.workspace = true 
# authors.workspace = true

例如,crate需要统一版本号,无需在每个crate写版本号。只需在[package]中写入version.workspace = true就能继承workspace中的版本号。升级版本号只需修改[workspace.package]的版本号,就同步了所有的crate版本。

类似的配置还有下面这些:
|||
|---|---|
|authors|categories|
|description|documentation|
|edition|exclude|
|homepage|include|
|keywords|license|
|license-file|publish|
|readme|repository|
|rust-version|version|

需要注意的是,这些配置项都需要定义在[workspace.package]配置块中。

2. 共享的依赖项

如同继承[workspace.package]配置,子包也能从[workspace.dependencies] 中继承dependencies。

/osk-json-lib/Cargo.toml :

[package] 
name = "bar" 
version.workspace = true 

[dependencies] 
regex = { workspace = true, features = ["unicode"] } 

[build-dependencies] 
cc.workspace = true 

[dev-dependencies] 
rand.workspace = true

继承依赖有两种写法:

# 从workspace中继承cc
1. cc.workspace = true 
# 很明显这种写法可以仅引入需要的features,而无需引入整个依赖项。
2. regex = { workspace = true, features = ["unicode"] }

3. Workspace可同时作为crate

我们前面讨论的把workspace当作组织多个crate的容器来使用。

根工程本身也可以作为一个crate,这种用法通常是子包都是lib,根工程是bin类型的,这样就可以看作是一个项目下细分了不同的子模块

如下,我们在[workspace]配置快的上面加入[package]配置快,那么这个根工程就被当作一个crate来看待了。假如这个项目是一个桌面软件,有两个子模块分别是video、audio。

[package]
name = "osk-desktop"
version = "0.1.0"
edition = "2021"

[workspace]
members = [
    "video",
    "audio"
]

Cargo.toml同时含有[package] 和 [workspace] 被称作 Root Package。开始介绍的用法被称作虚拟清单( virtual manifest )。这是Rust文档给这两种用法的命名,最重要的是理解这两种用法的本质区别。

总结

  • [workspace] — 定义工作空间。
    • resolver — 设置要使用的依赖解析器。
    • members — 要包含在工作区中的包。
    • exclude — 要从工作区中排除的包。
    • default-members — 当没有选择特定包时要操作的包。
  • [workspace.package] — 在包中继承的配置项。
  • [workspace.dependencies] — 用于继承包依赖项的依赖项。
  • [workspace.lints] — 在lints中继承的配置项。
  • [workspace.metadata] — 额外设置。
  • [patch] — 覆盖依赖项。
  • [replace] — 覆盖依赖项(deprecated)。
  • [profile] — 编译器设置和优化。

具体用法示例:https://github.com/oskwg/rust-workspace

相关文章
|
5月前
|
Rust 区块链
学Rust不学Cargo,等于没学Rust:features特性详解
在 Rust 中,Cargo 的 "features" 是一种条件编译机制,允许在编译 crate 时编译部分代码。这样可以在一个 crate 中提供多个功能,并根据需要选择性地启用或禁用这些功能。
180 1
|
28天前
|
Rust 编译器 测试技术
30天拿下Rust之深入Cargo
30天拿下Rust之深入Cargo
23 0
|
2月前
|
Rust 编译器 开发者
Cargo:Rust的神秘助手,它将如何改变包管理游戏规则?
【8月更文挑战第31天】Rust的包管理器Cargo简化了依赖管理和构建过程,与编译器无缝集成,提供从依赖下载到编译构建的全套解决方案。通过`cargo new`创建项目后,编辑`Cargo.toml`文件即可轻松管理依赖。Cargo还支持自动生成文档、运行测试及发布代码,并通过`crates.io`平台方便查找和分享Rust库,是Rust生态系统中的重要工具,有助于提升开发者生产力。
49 1
|
3月前
|
Rust 测试技术 编译器
Rust与C++的区别及使用问题之Rust项目中组织目录结构的问题如何解决
Rust与C++的区别及使用问题之Rust项目中组织目录结构的问题如何解决
|
4月前
|
Rust Shell 索引
使用阿里云镜像加速Rust与Cargo安装及更新
使用阿里云镜像加速Rust与Cargo安装及更新
758 0
|
4月前
|
Rust Unix Windows
使用Cargo国内镜像提升Rust开发效率
使用Cargo国内镜像提升Rust开发效率
375 0
|
4月前
|
Rust
使用Cargo创建、编译与运行Rust项目
使用Cargo创建、编译与运行Rust项目
159 0
|
5月前
|
Rust 测试技术 开发工具
Rust中的Cargo:依赖管理与项目构建
本文将深入探讨Rust编程语言中的Cargo工具。Cargo不仅用于构建Rust项目,还是管理项目依赖的关键组件。我们将了解如何使用Cargo创建新项目、添加依赖项、以及如何通过Cargo进行构建和测试,从而确保项目的顺利开发与部署。
|
5月前
|
Rust 开发工具 git
【一起学Rust】Rust包管理工具Cargo初步了解
【一起学Rust】Rust包管理工具Cargo初步了解
157 0
|
2月前
|
Rust 安全 Go
揭秘Rust语言:为何它能让你在编程江湖中,既安全驰骋又高效超车,颠覆你的编程世界观!
【8月更文挑战第31天】Rust 是一门新兴的系统级编程语言,以其卓越的安全性、高性能和强大的并发能力著称。它通过独特的所有权和借用检查机制解决了内存安全问题,使开发者既能享受 C/C++ 的性能,又能避免常见的内存错误。Rust 支持零成本抽象,确保高级抽象不牺牲性能,同时提供模块化和并发编程支持,适用于系统应用、嵌入式设备及网络服务等多种场景。从简单的 “Hello World” 程序到复杂的系统开发,Rust 正逐渐成为现代软件开发的热门选择。
56 1