首页 > 其他分享 >rust 代码组织结构

rust 代码组织结构

时间:2023-09-25 19:13:24浏览次数:44  
标签:组织 代码 crate Crate pub visible rust fn mod

使用包、Crate 和模块管理不断增长的项目 - Rust 程序设计语言 中文版

rust 组织结构中,包括以下几个概念

Package(包),Crate(箱),Moudle(模块)

  • Package

这是 Cargo 的概念,对应一个 Cargo.toml 文件,也就是一个 rust 工程。用于构建、测试、共享 Crate。
1 package = 0/1 lib crate + 0/N bin crate

  • Crate

有 bin / lib 之分,bin 是可执行 Crate, lib 是库 Crate。

基础概念 - bin Crate

通过 cargo new project-name 新建一个 cargo 项目之后,默认新建的是 bin Crate. 代码结构如下

src
  ╰-main.rs
Cargo.toml
Cargo.lock

默认约定,main.rs 表示的是 bin Crate,里面有 main 函数入口,其 Crate 的名称(也是产生的可执行文件的名称)就是项目名称。
使用 cargo run,默认运行的就是这个 Crate.

可以改吗?可以,但通常不会这么做。

如图所示,这里没有默认的 main.rs,在 Cargo.toml 文件中显式配置了两个 bin Crate. 名字分别是 cat1 和 cat2(对应的可执行文件的名字也是这个)。

文档见:Cargo Targets - The Cargo Book

此时运行 cargo run 就会有报错

error: `cargo run` could not determine which binary to run. Use the `--bin` option to specify a binary, or the `default-run` manifest key.
available binaries: cat1, cat2

需要这要运行 cargo run --bin cat1

根据默认的约定,如果有多个 bin Crate,除了 main.rs,其它的应该放在 bin 目录下。
如果需要 cargo run 命令能够找到默认的 Crate,则需要配置 default-run

这里配置 default-run = "cat", 其中的 cat 就是 main.rs 所指的 Crate,因为默认情况下,main.rs 的 Crate 名称就是项目(package)的名称。

bin 中其它 Crate 的名称,默认就是文件名。

为什么一个项目(package)中,可以有多个 bin 类型的 Crate,场景是什么?

// from GPT-4
在 Rust 中,一个 package 可以包含多个 bin(binary)类型的 crate,这种特性在一些场景中非常有用。

命令行工具套件:如果你正在开发一套命令行工具,每个工具都有自己的功能,但它们共享一些公共的库代码。在这种情况下,你可以为每个工具创建一个单独的 bin crate,然后将公共代码放在 lib crate 中。

多目标项目:如果你的项目需要生成多个不同的二进制文件,比如一个客户端和一个服务器,或者一个前端和一个后端,你可以为每个目标创建一个 bin crate。

示例代码:如果你正在创建一个库并且想要提供一些示例代码,你可以创建一个或多个 bin crate 来展示如何使用你的库。

在这些情况下,所有的 bin crate 都可以在同一个 package 中管理,这样可以共享版本号、依赖和构建设置,使得项目的管理变得更加方便。

基础概念 - lib Crate

// from GPT-4
在 Rust 中,一个 package 中只能有一个 library crate 的原因主要是为了简化编译和依赖管理的过程。在 package 中,library crate 的名字与 package 的名字相同,这使得在使用 cargo build 或 cargo publish 等命令时,cargo 可以明确知道应该编译或发布哪个 crate。

如果一个 package 中允许有多个 library crate,那么在处理依赖关系时,可能会产生一些复杂的问题。比如,如果两个 library crate 有相同的名字但版本不同,那么在解析依赖关系时就会产生冲突。另外,如果一个 package 中有多个 library crate,那么在编译时,cargo 也需要知道应该编译哪个 crate,这将使得编译过程变得更复杂。

总的来说,限制一个 package 中只有一个 library crate 是为了简化编译和依赖管理的过程,使得 Rust 的 package 管理系统更易于使用。

默认约定是,lib Crate,以 lib.rs 的文件名,直接放在 src 目录下。
或者可以通过 Crago.toml 自定义,文档:https://doc.rust-lang.org/cargo/reference/cargo-targets.html?highlight=[[bin]]#configuring-a-target

[lib]
name = "foo"           # The name of the target.
path = "src/lib.rs"    # The source file of the target.
test = true            # Is tested by default.
doctest = true         # Documentation examples are tested by default.
bench = true           # Is benchmarked by default.
doc = true             # Is documented by default.
plugin = false         # Used as a compiler plugin (deprecated).
proc-macro = false     # Set to `true` for a proc-macro library.
harness = true         # Use libtest harness.
edition = "2015"       # The edition of the target.
crate-type = ["lib"]   # The crate types to generate.
required-features = [] # Features required to build this target (N/A for lib).

同一个项目(package)中,bin Crate 可以直接引用 lib 中定义的内容,不需要在 Cargo.toml 中做额外的声明。

Module

从模块的角度理解 Crate,一个 Crate 就是一棵 Module 组成的树。其中树的根节点(一个隐式根节点)的名称,就是 crate

// lib.rs
pub fn add(a: i32, b: i32) -> i32 {
    a + b
}

pub mod sub_module {
    pub fn calc(a: i32, b: i32) -> i32 {
        crate::add(a, b)
    }
}

比如在 lib.rs 中定义了 add 这个函数,就是在 crate 这个隐式的根 Module 下面,可以使用 create::add 的方式,引用到这个函数。

在上面的例子中,除了使用隐式的根名称 crate,也可以使用 super 关键字

// lib.rs
pub fn add(a: i32, b: i32) -> i32 {
    a + b
}

pub mod sub_module {
    pub fn calc(a: i32, b: i32) -> i32 {
        super::add(a, b)
    }
}

super 指的是上一级。

Module 多文件

当然,通常情况下,不会在 lib.rs 中直接写具体的实现,而是使用 lib.rs 来进行统一的导出。

如上图中,在 math 子文件夹下,有一个 arithmetic.rs 文件,这里,math 和 arithmetic 就直接被视为两个模块,arithmetic 是 math 的子模块。

lib.rs 中,使用如下代码进行导出。

pub mod math {
    pub mod arithmetic;
}

在 main.rs 中进行使用

use cat::math;

fn main() {
    let sum = math::arithmetic::add(1, 2);

    println!("sum = {}", sum);
}

在 jump.rs 中进行使用

use cat::math::arithmetic;

fn main() {
    let sum = arithmetic::add(1, 2);

    println!("Hello, jump {}", sum);
}

pub

pub 就是 public, 所有声明的模块,函数,struct 等,默认都是私有的,都是使用 pub 将其公开。
如何仅在 Crate 内公开,而不在 Crate 外公开呢?简单的做法:在 lib.rs 中不导出就可以了。

pub 还有更精细的控制:
Visibility and privacy - The Rust Reference

  • pub(in path) makes an item visible within the provided path. path must be an ancestor module of the item whose visibility is being declared.
  • pub(crate) makes an item visible within the current crate.
  • pub(super) makes an item visible to the parent module. This is equivalent to pub(in super).
  • pub(self) makes an item visible to the current module. This is equivalent to pub(in self) or not using pub at all.
pub mod outer_mod {
    pub mod inner_mod {
        // This function is visible within `outer_mod`
        pub(in crate::outer_mod) fn outer_mod_visible_fn() {}
        // Same as above, this is only valid in the 2015 edition.
        pub(in outer_mod) fn outer_mod_visible_fn_2015() {}

        // This function is visible to the entire crate
        pub(crate) fn crate_visible_fn() {}

        // This function is visible within `outer_mod`
        pub(super) fn super_mod_visible_fn() {
            // This function is visible since we're in the same `mod`
            inner_mod_visible_fn();
        }

        // This function is visible only within `inner_mod`,
        // which is the same as leaving it private.
        pub(self) fn inner_mod_visible_fn() {}
    }
    pub fn foo() {
        inner_mod::outer_mod_visible_fn();
        inner_mod::crate_visible_fn();
        inner_mod::super_mod_visible_fn();

        // This function is no longer visible since we're outside of `inner_mod`
        // Error! `inner_mod_visible_fn` is private
        //inner_mod::inner_mod_visible_fn();
    }
}

fn bar() {
    // This function is still visible since we're in the same crate
    outer_mod::inner_mod::crate_visible_fn();

    // This function is no longer visible since we're outside of `outer_mod`
    // Error! `super_mod_visible_fn` is private
    //outer_mod::inner_mod::super_mod_visible_fn();

    // This function is no longer visible since we're outside of `outer_mod`
    // Error! `outer_mod_visible_fn` is private
    //outer_mod::inner_mod::outer_mod_visible_fn();

    outer_mod::foo();
}

fn main() { bar() }

use

将其它模块、struct、enum、函数等的定义,引入到当前作用域。
惯用做法:
引入函数时,只引入到函数的上层模块,然后通过 xxx::func() 的形式调用,函数归属会比较清晰;
引入 struct、enum 时,就全路径直接引入。

  • as 为引入的路径取别名。
use std::fmt::Result;
use std::io::Result as IoResult;

fn f1() -> Result {}
fn f2() -> IoResult {}
  • pub use

使用 pub use 可以将与引用到的内容重新导出,这个过程中,就有机会对导出做出调整

比如上例中的 lib.rs

// lib.rs
mod math {
    pub mod arithmetic;
}

pub use math::arithmetic;  // 直接导出 arithmetic 模块
// main.rs
use cat::arithmetic; // 直接引入 arithmetic 模块,这里没有 math 了

fn main() {
    let sum = arithmetic::add(1, 2);

    println!("sum = {}", sum);
}
  • 嵌套路径
use std::cmp::Ordering;
use std::io;

use std::{cmp::Ordering, io};
use std::io;
use std::io::Write;

use std::io::{self, Write};
  • 通配符全部引入
use std::collections::*;
  • 引入第三方库
[dependencies]
rand = "0.7.0"

Dependency Resolution - The Cargo Book

高级特性 Workspace

Cargo 的 Workspace 机制,通常一个项目就是一个 package,但当项目足够复杂时,一个项目中可以有多个 package,组成一个 Workspace。

案例:
wasmerio/wasmer:

标签:组织,代码,crate,Crate,pub,visible,rust,fn,mod
From: https://www.cnblogs.com/jasongrass/p/17728205.html

相关文章

  • 低代码引擎 TinyEngine 正式发布!
    在当今数字化飞速发展的时代,企业对高效、敏捷的应用程序需求日益旺盛。为了满足这一需求,越来越多的低代码开发平台开始涌现。这些平台通过提供简单易用的开发工具和优化后的开发流程,帮助开发者快速构建高质量、可重复使用的应用程序,同时降低了开发的难度和成本,提高了开发效率和灵活......
  • pytorch(3-2) 多层 线性回归 训练和预测代码
     脱离网页化python没有可视化    #%matplotlibinlineimporttorchimporttorchvisionfromtorch.utilsimportdatafromtorchvisionimporttransformsfromd2limporttorchasd2l#通过ToTensor实例将图像数据从PIL类型变换成32位浮点数格式,#并除以2......
  • 代码混淆和加固,保障应用程序的安全性
    ​ 前言iOS加固保护是直接针对iosipa二进制文件的保护技术,可以对iOSAPP中的可执行文件进行深度混淆、加密。使用任何工具都无法逆向、破解还原源文件。对APP进行完整性保护,防止应用程序中的代码及资源文件被恶意篡改。IpaGuard通过修改ipa文件中的macho文件中二进制数......
  • 代码混淆和加固,保障应用程序的安全性
    ​ 前言iOS加固保护是直接针对iosipa二进制文件的保护技术,可以对iOSAPP中的可执行文件进行深度混淆、加密。使用任何工具都无法逆向、破解还原源文件。对APP进行完整性保护,防止应用程序中的代码及资源文件被恶意篡改。IpaGuard通过修改ipa文件中的macho文件中二进制数......
  • 低代码引擎 TinyEngine 正式发布!
    在当今数字化飞速发展的时代,企业对高效、敏捷的应用程序需求日益旺盛。为了满足这一需求,越来越多的低代码开发平台开始涌现。这些平台通过提供简单易用的开发工具和优化后的开发流程,帮助开发者快速构建高质量、可重复使用的应用程序,同时降低了开发的难度和成本,提高了开发效率和灵......
  • 口算题卡代码
    8.(简答题)请完善课上的口算题卡代码,实现重复题目的检测、题目数字范围、加减乘除算式的参数化等扩展功能,提交代码和运行截图。packageorg.example;importjava.util.Random;importjava.util.Scanner;publicclassMain{publicstaticvoidmain(String[]args)......
  • 【遥遥领先】Eolink IDEA 插件:零代码入侵,自动生成接口
    省流版:Eolink有IDEA插件吗?有,而且遥遥领先!我们在一年半之前就发布了,而且功能更丰富!IDEA插件市场搜索“EolinkApikit”即可安装使用。......
  • 如何在低代码平台中应用可视化编程
    可视化编程,亦即可视化程序设计:以“所见即所得”的编程思想为原则,力图实现编程工作的可视化,即随时可以看到结果,程序与结果的调整同步。可视化编程的理念来源于可视化技术,它指的是一种把计算机程序中的文本指令转换为用户可以完全理解和操作的图形化界面。传统上,用户通过文本编程来......
  • Rust函数与闭包
    1.常规函数函数都拥有显示的类型签名,其本身也是一种类型。1.1函数类型自由函数//自由函数fnsum(a:i32,b:i32)->i32{a+b}fnmain(){assert_eq!(3,sum(1,2))}关联函数与方法structA(i32,i32);implA{//关联函数fnsum(a:i32,b:......
  • 《梦断代码》阅读笔记01
    1、与其他的书籍很不同的一点是:这本书有第0章而第0章有这么一句话,也是将我这两年来学习技术的心理状态给描绘了个大概:“helloworld”程序一无所用,但足以蛊惑人心,多少软件雄心勃勃,但最终未结善果。不得不承认的一点是,我当初刚开始使用IDEA编程工具学习Java的时候,坚持学习下去......