为自己量身打造一个 Rust 项目模板/脚手架

news2025/1/19 19:29:45

摘要

quick-start-rs(quick start a rust project)是用于快速创建一个 rust 项目的脚手架/模板。

  • 标题:为自己量身打造一个 Rust 项目模板/脚手架
  • 深度参考 Rust Code Quick Start
  • 文章来自 suhanyujie
  • Tags: Rust, utils, quick start, project template,脚手架

正文

当你心血来潮,想用 Rust 写一个小工具时,也许你可以直接使用 cargo new pro1001 之类的命令进行快速创建,但这样你需要做一些前置准备工作,比如:创建 utils crate、错误处理等等。现在也许你可以有更好的方式 —— quick-start-rs,当然,本文只是抛砖引玉,提供一个思路,你完全可以根据自己的需要定制自己的“quick-start-rs”。此外,本文也是参考 Rust Code Quick Start 撰写的。

本文需要你已经了解使用 VSCode + RA 进行开发 rust 项目开发。

创建项目

当然,从 0 到 1,我们还是使用 cargo new quit-start-rs 创建项目,并进入项目目录。

$ cargo new quit-start-rs
$ cd quit-start-rs

.
├── Cargo.lock
├── Cargo.toml
├── README.md
├── src
│  └── main.rs
└── target

utils crate

写项目时,经常会用到所谓的 utils 工具包,其中可能会无所不包,如字符串处理、加解密,以及一些 helper 方法。有一个快捷方式,直接在 main.rs 顶部加入 mod utils

mod utils; // <- New

fn main() {
    println!("Hello, world!");
}

此时 utils crate 尚未被创建和声明,我们只需将光标放置在报错处,按快捷键 alt + Enter(或者点击小灯泡):

此时 RA 会帮你创建好对应的目录和文件:

.
├── Cargo.lock
├── Cargo.toml
├── README.md
├── src
│  ├── main.rs
│  └── utils        //  <- New
│     └── mod.rs
└── target

此时就能直接在 ./src/utils/mod.rs 文件中编写工具函数。如果还需细分,可以在 src/utils 目录下继续创建文件或目录,用于更详细的模块划分。

错误处理

用 Rust 编写工具或项目时,错误处理是不可缺少的东西,随着项目的增加,引入其他第三方 crate,或有各种各样的错误类型,为了在你的项目中统一这些错误,我们可以用到 thiserror。所以我们可以将其加入的项目模板中:

[dependencies]
thiserror = "1"

在 main.rs 文件顶部,增加 mod errors;,然后根据“灯泡”提示,直接创建文件:

.
├── Cargo.lock
├── Cargo.toml
├── README.md
├── src
│  ├── errors.rs  // <- New
│  ├── main.rs
│  └── utils
│     └── mod.rs
└── target

然后在其中声明你自己项目的错误类型:

#[derive(thiserror::Error, Debug)]
pub enum Error {
    #[error("Generic {0}")]
    Generic(String),

    // // 例如将标准库中的 io error 统一为你定义的错误类型。以此类推,你可以将其他库中的错误统一为你声明的错误类型。
    #[error(transparent)]
    IO(#[from] std::io::Error), 
}

prelude

每个项目中都可以根据需要定制化地加入要引入的定义或者库,我们可以将这种预引入独立到一个文件中 prelude.rs。还是一样的方法,在 main.rs 文件顶部,增加 mod prelude;,然后根据“灯泡”提示,直接创建文件:

.
├── Cargo.lock
├── Cargo.toml
├── README.md
├── src
│  ├── errors.rs 
│  ├── main.rs  
│  ├── prelude.rs   // <- New
│  └── utils
│     └── mod.rs
└── target

可以在文件中加入需要的“预引入”,比如常用的 Result 别名定义:

// prelude.rs
use crate::errors::Error;

pub type Result<T> = std::result::Result<T, Error>;

这样,整个项目中,在需要返回 Result 类型时,直接在函数签名中定义如下的返回值即可(记得要引入 prelude use crate::prelude::*;):

use crate::prelude::*;

pub fn lib1() -> Result<String> {
    Ok("lib1 is called...".into())
}

New Type 模式

New Type 在 Rust 中是一种惯用法,可以将一个类型进行包装。使用它的意义可以参考 Rust 语言圣经中:

  • 为外部类型实现 trait
  • 更好的可读性
  • 隐藏内部细节

基于此,我们可以为该项目实现一个通用的 wrapper 类型,使其在你需要的时候,可以为你所使用:

// prelude.rs
pub struct W<T>(pub T);

示例 —— 读取目录下的文件

为了使用更好的理解,这里使用一个示例,读取指定目录下的文件列表。在一般情况下,我们可以使用如下代码实现:

/// read file list in some dir
fn get_files_by_dir(dir: String) -> Result<Vec<String>> {
    let mut list = vec![];
    for entry in fs::read_dir(dir)?.filter_map(|item| item.ok()) {
        let entry = entry
            .path()
            .to_str()
            .map(String::from)
            .ok_or_else(|| Error::Generic("Invalid path {entry:?}".into()))?;
        list.push(entry);
    }
    return Ok(list);
}

基于 fs::read_dir() 的迭代器,迭代的结果是 DirEntry,我们需要将其转换为 String 类型,我们可以将 DirEntry 用 W<T>(pub T) 包装一下,并为 W 类型实现 TryFrom trait:

// src/utils/dir_entry_from.rs
impl TryFrom<W<&DirEntry>> for String {
    type Error = Error;

    fn try_from(value: W<&DirEntry>) -> Result<String> {
        value
            .0
            .path()
            .to_str()
            .map(String::from)
            .ok_or_else(|| Error::Generic(format!("Invalid path: {:?}", value.0)))
    }
}

此时,get_files_by_dir 函数的实现就会简单干净很多:

fn get_files_by_dir(dir: String) -> Result<Vec<String>> {
    let mut list = vec![];
    for entry in fs::read_dir(dir)?.filter_map(|item| item.ok()) {
        let entry: String = W(&entry).try_into()?;
        list.push(entry);
    }
    return Ok(list);
}

此时可能会有同学问,为啥不直接为 DirEntry 实现 TryFrom trait,而非要用 W 类型包装一下?

主要原因是 Rust 中存在一种“孤儿规则”(orphan rule)导致,因为它,我们不能为非本地类型实现非本地的 trait,在这里,DirEntry 是标准库中的类型,TryFrom 也是标准库中的 trait,因此我们需要将 DirEntry 包装成一种新的本地类型 —— 即用 W<pub T> 对其进行包装,从而实现 TryFrom trait。

cargo generate

完成以上步骤后,你的项目模板基本成形,此时可以提交到 Github,然后基于这个仓库进行新项目的创建。基于模板创建新项目还需要一个工具 —— cargo generate,如果还没有安装,可以使用 cargo install cargo-generate 进行安装。

安装完成后,可以使用命令:cargo generate --git https://github.com/suhanyujie/quick-start-rs.git 进行拉取。

cargo-generate 默认是从 Github 上拉去仓库模板的,因此可以使用简短一点的方式:

cargo generate suhanyujie/quick-start-rs

然后按照命令行的提示,输入新项目名称即可生成(例如这里我输入的项目名是 demo1):

$  cargo generate suhanyujie/quick-start-rs
⚠️   Favorite `suhanyujie/quick-start-rs` not found in config, using it as a git repository: https://github.com/suhanyujie/quick-start-rs.git
🤷   Project Name: demo1
🔧   Destination: /some/path/xxx/demo1 ...
🔧   project-name: demo1 ...
🔧   Generating template ...
[ 1/15]   Done: .gitignore
[ 2/15]   Done: Cargo.lock
[ 3/15]   Done: Cargo.toml
[ 4/15]   Done: README.md
[ 7/15]   Done: docs/images
[ 8/15]   Done: docs
[ 9/15]   Done: src/errors.rs
[10/15]   Done: src/main.rs
[11/15]   Done: src/prelude.rs
[12/15]   Done: src/utils/dir_entry_from.rs
[13/15]   Done: src/utils/mod.rs
[14/15]   Done: src/utils
[15/15]   Done: src
🔧   Moving generated files into: `/some/path/xxx/demo1`...
💡   Initializing a fresh Git repository
✨   Done! New project created /some/path/xxx/demo1

实际上 cargo generate 功能很多,更详细的用法和配置可以参考官方仓库和文档。

结语

好了到这里,我们的项目脚手架的搭建和使用基本完成,我们向其中添加了一些基本的错误定义,工具库,以及预引入功能,可以让你在写从 0 到 1 的项目时,更快地聚焦于项目本身的逻辑,提高效率。如果你觉得有更好的方式方法,欢迎在 issues 中提问题交流 : )。

参考

  • 关于基于模板创建新项目可以参考 https://rust-github.github.io/
  • Rust Code Quick Start
  • cargo generate

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

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

相关文章

视频编解码 — 带宽预测

目录 一、带宽预测作用 二、基于延时的带宽预测算法&#xff1a; 1、算法流程 2、计算延时 3、Trendline Filter 4、网络状态判断 5、带宽调整更新 三、基于丢包的带宽预测算法 1、带宽调整 四、最大带宽探测算法 五、最终的预测选择 一、带宽预测作用 控制音视频发…

荧光AF染料477876-64-5,Alexa Fluor 532,AF532 NHS ester

●中文名&#xff1a;AF532 活性酯&#xff0c;AF5 532酯 ●英文名&#xff1a;AF532-NHS &#xff0c;AF532 NHS ester&#xff0c;Alexa Fluor 532 ●外观以及性质&#xff1a; 荧光AF染料是具有磺化基团的荧光物质。AF染料带负电荷&#xff0c;具有亲水性高、亮度高、光稳定…

软件和硬件中的调用

文章目录**1 概述****2.1 程序进程内的调用&#xff1a;函数调用****2.2 程序进程间的调用&#xff1a;IPC****2.3 远程程序调用&#xff1a;RPC****2.4 远程调用REST****3 硬件“调用”****3.1 综述****总线模型****3.2 片内的总线****3.3 Chiplet多DIE封装互联总线****3.4 板…

1.1 统计学习方法的定义与分类

统计学习方法的定义与分类统计学习的概念统计学习的定义统计学习运用到的领域统计学习的步骤统计学习的分类统计学习的概念 统计学习的定义 统计学习 (Statistical Machine Learning) 是关于计算机基于数据构建概率统计模型并运用模型对数据进行预测与分析的一门学科。 以计…

Python任务调度框架Rocketry

文章目录定时任务库对比简介与其余框架的区别安装初试调度器基础测试方法字符串格式具体时间间隔周期某时间段条件 API条件逻辑方法对比执行选项在主进程和线程中执行进程线程异步设置默认选项日志流水线在一个任务后执行输入作为输出会话级参数函数参数TODO&#xff1a;元参数…

为什么说继承是把双刃剑

为什么说继承是把双刃剑 继承其实是把双刃剑&#xff1a;一方面继承是非常强大的&#xff1b;另一方面继承的破坏力也是很强的。 继承广泛应用于各种Java API、框架和类库之中&#xff0c;一方面它们内部大量使用继承&#xff0c;另一方面它们设计了良好的框架结构&#xff0c…

20221130 RabbitMQ

MQ MQ&#xff08;Message Queue&#xff09;消息队列&#xff0c;是基础数据结构中“先进先出”的一种数据结构。一般用来解决应用解耦&#xff0c;异步消息&#xff0c;流量削峰等问题&#xff0c;实现高性能&#xff0c;高可用&#xff0c;可伸缩和最终一致性架构。 主要的…

【正点原子FPGA连载】第二十三章 DDS信号发生器实验摘自【正点原子】DFZU2EG/4EV MPSoC 之FPGA开发指南V1.0

1&#xff09;实验平台&#xff1a;正点原子MPSoC开发板 2&#xff09;平台购买地址&#xff1a;https://detail.tmall.com/item.htm?id692450874670 3&#xff09;全套实验源码手册视频下载地址&#xff1a; http://www.openedv.com/thread-340252-1-1.html 第二十三章 DDS信…

web网页设计期末课程大作业__公司官网 (曼谷科技 1页)带psd文件

⛵ 源码获取 文末联系 ✈ Web前端开发技术 描述 网页设计题材&#xff0c;DIVCSS 布局制作,HTMLCSS网页设计期末课程大作业 | 家公司官网网站 | 企业官网 | 酒店官网 | 等网站的设计与制 | HTML期末大学生网页设计作业&#xff0c;Web大学生网页 HTML&#xff1a;结构 CSS&…

基于CCE Kubernetes网络与持久化存储实战

✅作者简介&#xff1a; CSDN内容合伙人&#xff0c;全栈领域新星创作者&#xff0c;阿里云专家博主&#xff0c;阿里云问答板块版主&#xff0c;华为云享专家博主&#xff0c;掘金后端评审团成员 &#x1f495;前言&#xff1a; 最近云原生领域热火朝天&#xff0c;那么云原生…

2022年NPDP新版教材知识集锦--【第四章节】(4)

【初始设计与规格阶段】(全部获取文末) 产品设计规格旨在&#xff1a; 明确产品设计并提供量化和客观性; 将产品设计要求传达给设计团队的其他成员; 推进产品从设计到制造的开发。 1、功能性设计(DesignforFunctionality,DFF) 功能设计决定了产品的最终性能&#xff0c;功…

小猴吃苹果-第12届蓝桥杯Scratch选拔赛真题精选

[导读]&#xff1a;超平老师计划推出Scratch蓝桥杯真题解析100讲&#xff0c;这是超平老师解读Scratch蓝桥真题系列的第90讲。 蓝桥杯选拔赛每一届都要举行4~5次&#xff0c;和省赛、国赛相比&#xff0c;题目要简单不少&#xff0c;再加上篇幅有限&#xff0c;因此我精挑细选…

JCTC:基于PWmat中的混合溶剂模型精确计算离子溶解自由能

溶液环境中溶质离子或中间体的自由能计算是电化学研究中最棘手的问题之一。目前单纯的实验手段并不能对发生在溶液中的化学反应过程/机理进行直接探测&#xff0c;许多信息仍主要依赖于理论模拟。对于这一问题&#xff0c;目前很多研究者采用经验势场的溶剂模型方法&#xff0c…

Casein-PEG-Rhodamine B 络蛋白-聚乙二醇-罗丹明B Casein-RB

产品名称&#xff1a;络蛋白-聚乙二醇-罗丹明B 英文名称&#xff1a;Casein-PEG-Rhodamine B 质量控制&#xff1a;95% 原料分散系数PDI&#xff1a;≤1.05 存储条件&#xff1a;-20C&#xff0c;避光&#xff0c;避湿 用 途&#xff1a;仅供科研实验使用&#xff0c;不用于诊…

SAP 电商云 Spartacus UI Configurable Product 的页面设置

关键字 CPQ&#xff0c;Product Configuration&#xff0c;Product Configure&#xff0c;Product Variant 变体是在某些方面彼此不同但基于相同基本模型的产品。 变体的一个示例是 T 恤的颜色和尺寸。 在 Spartacus 中启用变体功能&#xff0c;并在 SAP Commerce Cloud 中配…

基于python的pulp库使用,从基础模型到复杂模型,从一维变量到二维变量

写在前面 学习笔记&#xff0c;仅作参考。 个人觉得配合步骤和建模&#xff0c;直接看代码就能入门pulp&#xff0c;所以没有啥解释&#xff0c;见谅。 参考 https://blog.csdn.net/youcans/article/details/116371416 步骤 1、安装PuLp &#xff08;pip install pulp) 2…

HRD特征及其检测方法简介

HRD特征及其检测方法简介1、HRD背景知识介绍1.1 HRR通路简介1.2 HRR基因突变可导致通路失活和HRD1.3 HRD高发癌种2、HRD的两类主要标志物2.1 致病基因2.2 基因组瘢痕3、HRD检测方法4、全景变异分析&#xff08;CGP&#xff09;5、关键信息6、参考文件1、HRD背景知识介绍 1.1 H…

(4)点云数据处理学习——其它官网例子

1、主要参考 &#xff08;1&#xff09;视频&#xff0c;大佬讲的就是好啊 【Open3D】三维点云python教程_哔哩哔哩_bilibili &#xff08;2&#xff09;官方的github地址 GitHub - isl-org/Open3D: Open3D: A Modern Library for 3D Data Processing &#xff08;3&#…

BUUCTF Misc 被偷走的文件 snake

被偷走的文件 下载文件 wireshark打开&#xff0c;搜索flag字符串 可以看到一个带有flag.rar的FTP包进行TCP流追踪 看来流量中有flag.rar&#xff0c;使用kali中的foremost进行文件分离 发现一个文件夹内有一个需要密码的压缩包 密码是5790&#xff0c;解压 得到flag …

Redis数据结构和类型

Redis 包含五种数据类型&#xff0c;分别为String、List、Hash、Set、ZSet 底层实现的数据结构包SDS、双向链表、压缩列表、哈希表、整数集合、跳表 redis结构图数据类型和数据结构的关系Redis六种数据结构 一、动态字符串(SDS) Redis 是用 C 语言实现的&#xff0c;但是它…