使用包、Crate 和模块管理项目(下)

1、使用 use 关键字将路径引入作用域

在之前的示例中我们引用模块中的函数或者结构体之类的,都是需要用到相对路径或者绝对路径去引用,然尔在这里,有一种方法可以简化这个过程。我们可以使用 use 关键字创建一个短路径,然后就可以在作用域中的任何地方使用这个更短的名字。

示例如下所示:

pub mod people {pub enum Sex {Man,Woman,}
}use crate::people::Sex;fn get_sex() {let man = Sex::Man;
}

在作用域中增加 use 和路径类似于在文件系统中创建软连接(符号连接,symbolic link)。通过在 crate 根增加 use crate::people::Sex,现在 Sex在作用域中就是有效的名称了,如同 Sex类型被定义于 crate 根一样。通过 use 引入作用域的路径也会检查私有性,同其它路径一样。

注意 use 只能创建 use 所在的特定作用域内的短路径。通过以下示例我们看一下。

pub mod people {pub enum Sex {Man,Woman,}
}
use crate::people::Sex;
mod ceshi {fn get_sex() {let man = Sex::Man;}
}

运行以下,看一下对应错误提示:

根据上图中所示:我们可以知道类型Sex没有被声明,所以use 声明的作用域和当前方法所在的作用域不同。如果在当前方法中进行引用,有2种方式:第一种就是通过super关键字去引用,use定义的值是在当前方法的父级种,第二种方式,就是use语句移动到当前模块种。

// 第一种方式
use crate::people::Sex;
mod ceshi {fn get_sex() {let man = super::Sex::Man;}
}
// 第二种方式
mod ceshi {use crate::people::Sex;fn get_sex() {let man = Sex::Man;}
}

1.1 创建惯用的 use 路径

在之前的示例当中,我们使用use的时候,直接引用到了具体的结构体类型定义,使用 use 引入结构体、枚举和其他项时,习惯是指定它们的完整路径。,示例如下所示:

pub mod root {pub mod people {pub enum Sex {Man,Woman,}}
}
use crate::root::people::Sex;
fn get_sex() {let man = Sex::Man;
}

如果是函数类型时,我们必须在调用函数时指定父模块,这样可以清晰地表明函数不是在本地定义的,同时使完整路径的重复度最小化。示例如下所示:

pub mod root {pub mod people {pub fn getPeo() {}}
}
use crate::root::people;
fn get_sex() {let man = people::getPeo;
}

这种习惯用法背后没有什么硬性要求:它只是一种惯例,人们已经习惯了以这种方式阅读和编写 Rust 代码。

这个习惯用法有一个例外,那就是我们想使用 use 语句将两个具有相同名称的项带入作用域,因为 Rust 不允许这样做。

use std::fmt;
use std::io;fn function1() -> fmt::Result {// --snip--Ok(())
}fn function2() -> io::Result<()> {// --snip--Ok(())
}

如你所见,使用父模块可以区分这两个 Result 类型。如果我们是指定 use std::fmt::Result 和 use std::io::Result,我们将在同一作用域拥有了两个 Result 类型,当我们使用 Result 时,Rust 则不知道我们要用的是哪个。这样就是我们指定到具体类型时,可能重名的几率太高,所以会引用到它的父级来区分就行。

1.2 使用 as 关键字提供新的名称

使用 use 将两个同名类型引入同一作用域这个问题还有另一个解决办法:在这个类型的路径后面,我们使用 as 指定一个新的本地名称或者别名。示例如下所示:

use std::fmt::Result;
use std::io::Result as ResultAli;fn function1() -> Result {// --snip--Ok(())
}fn function2() -> ResultAli<()> {// --snip--Ok(())
}

1.3 使用 pub use 重导出名称

使用 use 关键字,将某个名称导入当前作用域后,这个名称在此作用域中就可以使用了,但它对此作用域之外还是私有的。如果想让其他人调用我们的代码时,也能够正常使用这个名称,就好像它本来就在当前作用域一样,那我们可以将 pub 和 use 合起来使用。这种技术被称为 “重导出re-exporting)”:我们不仅将一个名称导入了当前作用域,还允许别人把它导入他们自己的作用域。

示例如下所示:

pub mod root {pub mod people {pub fn getPeo() {}}
}
pub use crate::root::people; // 重导出
fn get_sex() {let man = people::getPeo;
}

在这个修改之前,外部代码需要使用路径 my_project::root::people::getPeo()来调用 getPeo()函数。现在这个 pub use 从根模块重导出了 people模块,外部代码现在可以使用路径 restaurant::peopel::getPeo

1.4 使用外部包

在使用外部包时,在 Cargo.toml 中加入对应包名称以及对应的版本号,如下所示:

rand = "0.8.5"

在 Cargo.toml 中加入 rand 依赖告诉了 Cargo 要从 crates.io 下载 rand 和其依赖,并使其可在项目代码中使用。

接着,为了将 rand 定义引入项目包的作用域,我们加入一行 use 起始的包名,它以 rand 包名开头并列出了需要引入作用域的项。示例如下:

use rand::Rng;
fn main() {let secret_number = rand::thread_rng().gen_range(1..100);println!("res {}", secret_number)
}

cargo run运行一下会发现去下载对应的资源,然后再打印1到100之间的一个随机数如下所示:

1.5 嵌套路径来消除大量的 use 行

当需要引入很多定义于相同包或相同模块的项时,为每一项单独列出一行会占用源码很大的空间。示例如下所示:

use std::cmp::Ordering;
use std::io;

相反,我们可以使用嵌套路径将相同的项在一行中引入作用域。这么做需要指定路径的相同部分,接着是两个冒号,接着是大括号中的各自不同的路径部分,示例如下所示:

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

我们可以在路径的任何层级使用嵌套路径,这在组合两个共享子路径的 use 语句时非常有用。例如一下示例:

use std::io;
use std::io::Write;

两个路径的相同部分是 std::io,这正是第一个路径。为了在一行 use 语句中引入这两个路径,可以在嵌套路径中使用 self,示例如下所示:

use std::io::{self, Write};

1.6 通过 glob 运算符将所有的公有定义引入作用域

如果希望将一个路径下 所有 公有项引入作用域,可以指定路径后跟 *,glob 运算符:

use std::collections::*;

这个 use 语句将 std::collections 中定义的所有公有项引入当前作用域。使用 glob 运算符时请多加小心!Glob 会使得我们难以推导作用域中有什么名称和它们是在何处定义的。 

2、将模块拆分成多个文件 

前面的示例都是一个文件中定义多个模块。当模块变得更大时,你可能想要将它们的定义移动到单独的文件中,从而使代码更容易阅读。

例如在之前的 src/lib.rs 中的如下代码

pub mod garden {pub mod vegetables {#[derive(Debug)]pub struct Asparagus {pub color: String,pub number: i32,}}
}

然后对src/lib.rs文件种的内容进行改写,如下所示:

mod garden;use garden::vegetables::Asparagus;pub fn getAsparagus() {let res = Asparagus {color: String::from("red"),number: 12,};
}

首行声明了 mod garden;在src/garden.rs 文件种进行查找。内容如下所示:

pub mod vegetables;

 接着我们创建一个 src/garden 目录和一个包含 Asparagus结构体定义的 vegetables.rs文件:文件内容如下所示:

pub struct Asparagus {pub color: String,pub number: i32,
}

目录结构如下所示: 

my-project

├─ Cargo.lock

├─ Cargo.toml

├─ README.md

└─ src

   ├─ garden

   │  └─ vegetables.rs

   ├─ garden.rs

   ├─ lib.rs

   └─ main.rs

总结:

在引用模块的时候一般有两种方式:

第一种就是上面所示的src/garden/vegetables.rs

第二种方式方式就是src/garden.rs,vegetables模块写在当前文件中。

如果你对同一模块同时使用这两种路径风格,会得到一个编译错误。在同一项目中的不同模块混用不同的路径风格是允许的,不过这会使他人感到疑惑。

使用 mod.rs 这一文件名的风格的主要缺点是会导致项目中出现很多 mod.rs 文件,当你在编辑器中同时打开它们时会感到疑惑。

我们将各个模块的代码移动到独立文件了,同时模块树依旧相同。这样解耦性更强,更方便与后于模块的迁移和重构。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/235446.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

云原生扫盲篇

What 云原生加速了应用系统与基础设施资源之间的解耦,向下封装资源以便将复杂性下沉到基础设施层;向上支撑应用,让开发者更关注业务价值 云原生是一种构建和运行应用程序的方法,也是一套技术体系和方法论. Cloud 表示应用程序位于云中而不是传统的数据中心Native表示应用程序从…

【打卡】牛客网:BM68 矩阵的最小路径和

模板的&#xff1a; 调试很久的地方&#xff1a; 最小的路径和而不是最大路径和。dp[i][j] matrix[i][j] (dp[i-1][j] > dp[i][j-1]? dp[i][j-1]:dp[i-1][j]); //需要加上括号dp和matrix下标的对应。 class Solution { public:/*** 代码中的类名、方法名、参数名已经指定…

C# NPOI导出datatable----Excel模板画图表

1、创建Excel模板 2、安装NPOI管理包 3、创建工作簿 &#xff08;XLSX和XLS步骤一样&#xff0c;以XLS为例&#xff09; IWorkbook workbook null; string time DateTime.Now.ToString("yyyyMMddHHmmss"); string excelTempPath Application.StartupPath "…

hackrf OOK算法简单数据分析(傅里叶计算)

import numpy as npdef savedata():data []i 0c 0while c < 100:if c 50:data.append((127.5*np.sin(2*np.pi/(32000/1000)))) # 当数据为 1 时data.append((127.5*np.cos(2*np.pi/(32000/1000))))data.append((i*127.5*np.sin(2*np.pi/(32000/1000)))) # 当数据为 0…

Apache Pulsar 技术系列 - PulsarClient 实现解析

导语 Apache Pulsar 是一个多租户、高性能的服务间消息传输解决方案&#xff0c;支持多租户、低延时、读写分离、跨地域复制&#xff08;GEO replication&#xff09;、快速扩容、灵活容错等特性。同时为了达到高性能&#xff0c;低延时、高可用&#xff0c;Pulsar 在客户端也…

揭秘Python 3.12的功能更新

大家好&#xff0c;Python 3.12已于2023年10月2日发布&#xff0c;它带来了许多更新&#xff0c;提升了编程体验&#xff0c;让我们通过一些实际示例来了解主要的更新内容。 1. 增强的f-strings f-strings现在更加灵活&#xff0c;允许使用多行表达式和注释。 # 在Python 3.…

论文笔记 | Nature 2023 FunSearch:利用大语言模型在数学科学领域探索新的发现

文章目录 一、前言二、主要内容三、总结🍉 CSDN 叶庭云:https://yetingyun.blog.csdn.net/ 一、前言 科学中有许多难以解决的问题,这些问题难以获得确切解答,但却相对容易进行验证。在数学和计算机科学领域,这类问题被称为 NP 完全优化问题(NP-complete optimization pr…

快速从图中提取曲线坐标数据的在线工具WebPlotDigitizer

快速从图中提取曲线坐标数据的在线工具WebPlotDigitizer 1 介绍2 WebPlotDigitizer在线版的使用2.1 上传图像2.2 点击横纵坐标点&#xff1a;2.3 选择曲线 3 查看数据参考 1 介绍 写论文时要对比别人曲线图、点图、柱形图的数据&#xff0c;但是只有图没有原始数据怎么办&…

.env.development是什么

.env.development 是一个配置文件&#xff0c;用于存储开发环境中的环境变量和配置信息。这种命名约定常见于Web开发和应用程序开发中&#xff0c;特别是在使用JavaScript或Node.js等技术栈的项目中。 这个文件的名称中包含了两部分信息&#xff1a; .env&#xff1a;这是通用…

最新国内可用使用GPT4.0,GPT语音对话,Midjourney绘画,DALL-E3文生图

一、前言 ChatGPT3.5、GPT4.0、GPT语音对话、Midjourney绘画&#xff0c;相信对大家应该不感到陌生吧&#xff1f;简单来说&#xff0c;GPT-4技术比之前的GPT-3.5相对来说更加智能&#xff0c;会根据用户的要求生成多种内容甚至也可以和用户进行创作交流。 然而&#xff0c;GP…

【优化】XXLJOB修改为使用虚拟线程

【优化】XXLJOB修改为使用虚拟线程 新建这几个目录 类&#xff0c; 去找项目对应的xxljob的源码 主要是将 new Thread 改为 虚拟线程 Thread.ofVirtual().name("VT").unstarted 以下代码是 xxljob 2.3.0版本 举一反三 去修改对应版本的代码 <!-- 定…

计算机基础以及实施运维工程师的介绍

目录 什么是实施、运维工程师 实施工程师 实施工程师的职责 什么是运维工程师 运维功工程师的职责 需要的技术 计算机的介绍 CPU 存储器 IO 系统总线 主板 BIOS 什么是实施、运维工程师 实施工程师 纯实施工程师是指在工程项目实施阶段专门负责实施工作的工程师。与其他…

springboot(ssm失物招领平台 物品挂饰招领系统Java系统

springboot(ssm失物招领平台 物品挂饰招领系统Java系统 开发语言&#xff1a;Java 框架&#xff1a;ssm/springboot vue JDK版本&#xff1a;JDK1.8&#xff08;或11&#xff09; 服务器&#xff1a;tomcat 数据库&#xff1a;mysql 5.7&#xff08;或8.0&#xff09; 数…

大模型赋能“AI+电商”,景联文科技提供高质量电商场景数据

据新闻报道&#xff0c;阿里巴巴旗下淘天集团和国际数字商业集团都已建立完整的AI团队。 淘天集团已经推出模特图智能生成、官方客服机器人、万相台无界版等AI工具&#xff0c;训练出了自己的大模型产品 “星辰”&#xff1b; 阿里国际商业集团已成立AI Business&#xff0c;…

Gazebo GUI模型编辑器

模型编辑器 现在我们将构建我们的简单机器人。我们将制作一个轮式车辆&#xff0c;并添加一个传感器&#xff0c;使我们能够让机器人跟随一个斑点&#xff08;人&#xff09;。 模型编辑器允许我们直接在图形用户界面 &#xff08;GUI&#xff09; 中构建简单的模型。对于更复…

在使用mapstruct,想忽略掉List<DTO>字段里面的,`data` 字段的映射, 如何写ignore: 使用@IterableMapping

在使用mapstruct,想忽略掉List字段里面的,data 字段的映射, 如何写ignore 代码如下: public interface AssigmentFileMapper {AssigmentFileDTO assigmentFileToAssigmentFileDTO(AssigmentFile assigmentFile);AssigmentFile assigmentFileDTOToAssigmentFile(Assigment…

用全志R128复刻自平衡赛车机器人,还实现了三种不同的操控方式

经常翻车的朋友们都知道&#xff0c;能在翻车后快速摆正车身的车才是好车。 就像动画《四驱兄弟》中展现的那样&#xff0c;在比赛中需要跟着赛车一起跑圈&#xff0c;而且赛车如果被撞翻还需要重新用手扶正&#xff0c;所浪费的时间非常影响比赛结果。 如果小豪和小烈可以拥有…

flutter自定义地图Marker完美展示图片

世人都说雪景美 寒风冻脚无人疼 只道是一身正气 结论 参考Flutter集成高德地图并添加自定义Maker先实现自定义Marker。如果自定义Marker中用到了图片&#xff0c;那么会碰到图片没有被绘制到Marker的问题&#xff0c;此时需要通过precacheImage来预加载图片&#xff0c;从而解…

Codeforces Round 916 (Div. 3)(E:贪心 F贪心dfs G tarjan+topsort +线段树优化建图)

A&#xff1a;直接暴力统计每个字符的次数是否达标即可 #include<bits/stdc.h> using namespace std; const int N 3e510,mod998244353; #define int long long typedef long long LL; typedef pair<int, int> PII; typedef unsigned long long ULL;const long l…

智能化创作与艺术:发展、问题、未来趋势

导言 随着人工智能技术的不断进步&#xff0c;智能化创作在艺术领域逐渐崭露头角。本文将深入研究智能化创作与艺术的发展过程、遇到的问题、解决的过程&#xff0c;探讨未来的可用范围&#xff0c;并分析在各国的应用和未来的研究趋势。最后&#xff0c;探讨在哪些方面能取胜&…