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

2023-11-10

摘要

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
									
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

为自己量身打造一个 Rust 项目模板/脚手架 的相关文章

随机推荐

  • Ps如何制作动态图片

    制作动态图片 按操作慢慢来 下面是我们要使用的图片 0 首先我们新建文件 宽 500px 高 500px 1 之后我们简单的设计一下画面 美观一下 需要用的字也先一下 我的比较丑 2 之后重点来了 重点来了 重点来了 从菜单工具 gt 窗口
  • 大数据:频繁项集

    大数据 频繁项集 下面是我 下面是阅读 大数据 互联网大规模数据挖掘与分布式处理 一书第六章笔记 详细请见该书所述 1 购物篮数据 项与购物篮 多对多的关系 项存放于购物篮
  • Book I-IV of Power

    复杂度1 5 机密度3 5 最后更新2021 04 24 任何CPU都有自己的及相关的规范 这些规范用来协调跨公司的软硬件开发者 使用者 共同建设围绕该CPU的软硬件生态体系 Power CPU是IBM所有CPU最终集大成者 从最早的RIS
  • 线性代数(4)——特征值与二次型

  • Realtime_Multi-Person_Pose_Estimation训练问题

    https blog csdn net kkae8643150 article details 102711101 前言 最近在研究Realtime Multi Person Pose Estimation的训练和再训练的过程 参考 htt
  • element -ui table表格内容无限滚动 使用插件vue-seamless-scroll

    使用插件 一 安装组件依赖 npm install vue seamless scroll 二 引入组件 import vueSeamlessScroll from vue seamless scroll components vueSea
  • csdn积分获取攻略

    下载积分攻略 1 个人设置里进行手机绑定CSDN账户 奖励50分 右上角设置 账户安全 手机绑定 2 完成任务送若干分积分 http task csdn net 3 上传有效资源获取积分 上传非法 广告资源用户 将被扣除一定积分 严重者封号
  • matplotlib 画图总结

    1 图片基本设置 import matplotlib pyplot as plt 图片尺寸 plt figure width height 方式1 plt rcParams figure figuresize width height 方式
  • 导入spacy时报错OSError: [E050] Can‘t find model ‘en‘. It doesn‘t seem to be a shortcut link,

    报错如下 File home muli local lib python3 6 site packages spacy util py line 175 in load model raise IOError Errors E050 for
  • element-UI使用el-select做字典映射时label值不显示问题

    问题描述 在使用elementUI的el select组件时做了字典影射 但是在选择option选项后选择框内并没有选中的值出现 这是通过调试发现被绑定的值已经改变 进行别的操作更新完dom后发现选项更新 操作 点击选择test选项 此处是
  • 简单了解YOLOv8

    简单介绍YOLOv8 这里主要关注模型的backbone和后处理的过程 并通过对比YOLOv5的架构来更深入的了解YOLOv8 模型框架 YOLOv5中的C3替换为更精简的C2f 即增加了更多的跳跃连接和split操作 Backbone 中
  • uniapp 自定义标题情况下,让标题和右侧胶囊对齐

    实现效果 无论手机类型怎么切换 自定义标题始终跟胶囊平齐 实现 在pages json文件中配置标题自定义 在index vue页面 编写自定义的标题内容 在onLoad里可以计算高度
  • 【深度学习】入门理解ResNet和他的小姨子们(三)---ResNeXt

    文章名称 Aggregated Residual Transformations for Deep Neural Networks 文章链接 https arxiv org abs 1611 05431 其实ResNeXt这个网络结构严格说
  • 大规模流量下的云边端一体化流量调度体系

    火山引擎是字节跳动旗下的云服务平台 将字节跳动快速发展过程中积累的增长方法 技术能力和工具开放给外部企业 提供云基础 视频与内容分发 数智平台VeDI 人工智能 开发与运维等服务 帮助企业在数字化升级中实现持续增长 LiveVideoSta
  • 构建领域驱动的Java应用

    引言 在现代软件开发中 设计和构建复杂的应用程序是一项充满挑战的任务 为了更好地满足业务需求和提供可维护的代码 软件开发者需要采用一些强大的工具和技术 领域驱动设计 Domain Driven Design 简称DDD 是一种优秀的方法 它
  • Codeforces 1210 D Konrad and Company Evaluation —— 暴力

    This way 题意 现在有n个人 第i个人的工资一开始是i 现在有一些人相互讨厌 然后如果第x个人和第y个人相互讨厌 并且x的工资比y高 那么x就会向y炫耀 x y z这三个人的组合是危险的 当x会向y炫耀 y会向z炫耀 每次修改一个人
  • 用户消费行为分析

    消费品用户行为分析 根据CDNOW的一段用户订单数据进行消费行为分析 CDNow是一家在线音乐零售平台 后被德国波泰尔斯曼娱乐集团公司出资收购 其资产总价值在最辉煌时曾超过10亿美元 下面主要通过分析CDNow网站的用户购买明细来分析该网站
  • Kafka拉取某一个时间段內的消息

    一般来说我们都使用Kafka来记录用户的操作记录以便后续分析 但是通常使用的时候需要按天来统计每天的去重用户数 点击量之类的 这个时候如果直接拉某个topic的数据的话 就需要判断每个消息的时间戳 还要兼顾把所有的Partition都拉完才
  • 考试系统服务器考试机,考试系统

    考试系统为 B S 结构 考试中心需具备 Win2000 服务器且安装 IIS5 0 的软件环境和一定规模的局域网硬件环境 视参加考试的学员人数决定 客户端须安装 IE5 0 或以上浏览器版本 本系统从技术上充分考虑了考试过程的完整性和安全
  • 为自己量身打造一个 Rust 项目模板/脚手架

    摘要 quick start rs quick start a rust project 是用于快速创建一个 rust 项目的脚手架 模板 标题 为自己量身打造一个 Rust 项目模板 脚手架 深度参考 Rust Code Quick St
Powered by Hwhale