【Rust类型驱动开发 Type Driven Development】

Rust类型驱动开发

  • 什么是类型驱动开发
  • Rust类型驱动开发的实际应用
    • Rust中类型驱动开发的简单举例
    • Rust中类型驱动开发的更具体的示例
      • Rust 中类型驱动开发Api
  • 总结


什么是类型驱动开发

type-diven development,即将开发者需要的逻辑约束条件编码到类型系统中,举个例子,一般业务逻辑下开发者通常需要检查输入的正确性,一般情况下开发者的思维惯性会在调用function的内部做一些逻辑判断,比如字符串长度是否符合条件,格式是否符合如人名,邮箱地址等等。类型驱动开发是将此类检查放在形成某一种类型的时候完成,即如果逻辑不通则不会形成需要的类型,同样的,类型驱动开发也具备一定的封装性,保证开发者不会因为误用而破坏流程的安全性和产生一些垃圾数据。


Rust类型驱动开发的实际应用

Rust中类型驱动开发的简单举例

代码示例:

use unicode_segmentation::UnicodeSegmentation;struct Address(String);impl Address {pub fn TryParseEmail(s: String) -> Address {let is_empty = s.trim().is_empty();let is_too_long = s.graphemes(true).count() > 256;let forb_charc = ['/', '(', ')', '"', '<', '>']; //and so onlet is_legal = s.chars().any(|c| forb_charc.contains(&c));if (is_legal || is_empty || is_too_long) {panic!("{} is not a valid address", s);} else {Address(s)}}
}fn main() {let s: String = "asdfasdf//s".to_string();Address::TryParseEmail(s);
}

如上述代码,当程序收集到了一个地址字符串,TryPaseEmail尝试将它转换成一个Address类型,无法形成Address类型的字符串即可标记为不合法的地址,从而抛出错误中断线程。需要注意的是,这里使用的是结构体元组,也就是指定的类型,从而无需在所有需要地址的位置都去判断是否为空,是否合法等操作,这对于大型的程序来说十分友好。

Rust中类型驱动开发的更具体的示例

Rust 中类型驱动开发Api

源代码参考视频地址
此代码实现一个类似进度条的打印,通过trait限定可使用的范围,在编译期间便可抛出更加精准的错误,相较于C++的模板编程,只有在使用到了这段代码的程序才能发现这段代码的问题来讲更加的迅速和及时,而相较于python这种动态语言来讲,python给出的错误一般情况下会摸不着头脑或者既不以预期方式运行也没有报错,这样会使得开发者更加的头疼。
代码示例:

use std::{thread::sleep, time::Duration};struct Unbounded;
struct Bounded {bound: usize,delims: (char, char),
}
//上述Unbounded,Bounded即描述迭代器是否有边界,既大小是不是"有限的",有限则在实现的角度会添加括号已示边界。
struct Progress<Iter, Bound> {iter: Iter,i: usize,bound: Bound,
}
//Progress,进度条结构体, 包含迭代器,i表示进度条长度,bound表示边界,边界包含长度和用什么符号来表示边界trait ProgressDisplay: Sized {fn display<Iter>(&self, progress: &Progress<Iter, Self>);
}
//trait ProgressDisplay 需要trait Sized以便Progress在编译期大小可知,类型安全,否则无法通过编译
impl ProgressDisplay for Unbounded {fn display<Iter>(&self, progress: &Progress<Iter, Self>) {println!("{}", "|".repeat(progress.i))}
}
impl ProgressDisplay for Bounded {fn display<Iter>(&self, progress: &Progress<Iter, Self>) {println!("{}{}{}{}",progress.bound.delims.0,"|".repeat(progress.i)," ".repeat(self.bound - progress.i),progress.bound.delims.1);}
}
//分别为有界无界实现打印方式,主要区别,有边界的两边有符号,且长度可知,未遍历到的进度用空格填充
impl<Iter> Progress<Iter, Unbounded> {pub fn new(iter: Iter) -> Self {Progress {iter,i: 0,bound: Unbounded,}}
}
//默认无界
impl<Iter, Bound> Iterator for Progress<Iter, Bound>
whereIter: Iterator,Bound: ProgressDisplay,
{type Item = Iter::Item;fn next(&mut self) -> Option<Self::Item> {self.bound.display(&self);self.i += 1;self.iter.next()}
}
//为进度条实现迭代器,方可迭代,实现关键步骤:Item & next 方法
trait ProgressIteratorExt: Sized {fn progress(self) -> Progress<Self, Unbounded>;
}impl<Iter> ProgressIteratorExt for Iter
whereIter: Iterator,
{fn progress(self) -> Progress<Self, Unbounded> {Progress::new(self)}
}
//便于使用Dot(.progress())将容器初始化为进度条
//ExactSizeIterator implementation
impl<Iter> Progress<Iter, Unbounded>
whereIter: ExactSizeIterator,
{pub fn with_bound(self) -> Progress<Iter, Bounded> {let bound = Bounded {bound: self.iter.len(),delims: ('[', ']'),};Progress {i: self.i,iter: self.iter,bound,}}
}
//with bound 会加上默认边界,并将默认的无边界进度条转为有边界的
impl<Iter> Progress<Iter, Bounded> {pub fn with_delims(mut self, delims: (char, char)) -> Self {self.bound.delims = delims;self}
}fn main() {let brkt = ('<', '>');let vector = vec![1, 2, 3, 4, 5];for _ in vector.iter().progress().with_bound().with_delims(brkt) {sleep(Duration::from_secs(1));}//以下代码将会出错,错误的调用方法,使用无界progress调用需要有界进度条类型的方法。for _ in (0..).progress().with_delims(brkt) {sleep(Duration::from_secs(1));}
}

以上展示的是在开发者自行实现一些组件的时候,通过类型驱动开发可以进一步保证调用的正确性,常规方式用trait限定和明确的类型定义调用与之相关的方法,从而保证Api的安全稳定。


总结

以上便是Rust中的类型驱动开发的粗浅理解,类型驱动开发并不是一个新概念,而是通过Rust的语言特点可以更好的表达,并且在Rust编程中也很常见,有的开发者称之为新类型模式,这通常体现在,Api的入参验证,通过中间方法,如将string转换成特定的邮箱地址类型,人名类型等等,从而使得开发专注于逻辑本身而不是语言限定本身。

“苟日新,日日新,又日新,常看常新”

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

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

相关文章

【论文复现】基于标签相关性的多标签学习

&#x1f4dd;个人主页&#x1f339;&#xff1a;Eternity._ &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; ❀基于标签相关性的多标签学习 论文概述什么是多标签学习论文贡献 算法流程挖掘“主题“——提取标签相关性训练 &#x1d440; &#x1d447; …

MMCloud+JuiceFS:云端Nextflow工作流的新方案

在云计算和生物信息学领域&#xff0c;执行大规模计算任务时的性能与成本效率至关重要。MemVerge推出的JuiceFlow通过JuiceFS和云端Memory Machine Cloud&#xff08;简称“MMCloud”&#xff09;平台&#xff0c;为Nextflow工作流&#xff08;pipeline&#xff09;提供了一种高…

【Linux】 shell 学习汇总[转载]

转载地址&#xff1a;https://blog.csdn.net/baidu_33718858/article/details/81453835 一些平时使用过程中的知识点积累&#xff0c;来源都附上了博客&#xff0c;添加了一些自己的总结。 感触&#xff1a;linux命令用熟了相当提高工作效率&#xff0c;有时候用Python写十几行…

天云数据联手举办“科学传播沙龙”活动,探讨Sora是否会带来新的科学革命

4月18日&#xff0c;由北京市科协主办&#xff0c;北京科技记协承办&#xff0c;中关村创新研修学院、天云融创数据科技&#xff08;北京&#xff09;有限公司协办的“AIGC塑造数字内容生产新范式”科学传播沙龙在京举办&#xff0c;活动由北京市科协宣传文化部二级调研员、北京…

LlamaIndex+本地部署InternLM实践

1.环境配置 1.1 配置基础环境 这里以在 Intern Studio 服务器上部署 LlamaIndex 为例。 首先&#xff0c;打开 Intern Studio 界面&#xff0c;点击 创建开发机 配置开发机系统 填写 开发机名称 后&#xff0c;点击 选择镜像 使用 Cuda11.7-conda 镜像&#xff0c;然后在资源…

uni-segmented-control 分段器添加数量提示

1、 在 template 内写入 uni-segmented-control 基础写法 <uni-segmented-control :current"current" :values"items" :style-type"styleType" clickItem"onClickItem" activeColor"#81D8D0" />2、在 script 里面写…

Linux篇(用户管理命令)

目录 一、用户与用户组 1. 为什么要做用户与用户组管理 2. Linux的用户及用户组 2.1. Linux的多用户多任务 2.2. 什么是用户 2.3. 什么是用户组 2.4. 用户和用户组的关系 二、用户和用户组管理 1. 用户组管理 1.1. 用户组添加 /etc/group文件结构 1.2. 用户组修改 …

编程之路,从0开始:内存函数

Hello大家好&#xff01;很高兴我们又见面了。 给生活添点passion&#xff0c;开始今天的编程之路&#xff01; 今天我们来讲C语言中的内存函数。 目录 1、memcpy内存复制 2、memmove可重叠内存拷贝 3、memset设置字符 4、memcmp比较 1、memcpy内存复制 memcpy就是内存复制…

PyCharm 中的【控制台】和【终端】的区别

pycharm专业版-使用 PyCharm 中的【控制台】和【终端】的区别如下&#xff1a; 1.环境&#xff1a;控制台是 PyCharm 的内部环境&#xff0c; 终端 是操作系统的命令行界面。 2.功能&#xff1a;控制台可以运行 Python 代码&#xff0c;并显示执行结果&#xff1b; 终端可以…

IDEA修改注释颜色—图文教程

老的注释颜色用习惯了&#xff0c;新电脑的灰色注释不习惯&#xff0c;还是喜欢黄色哈哈哈哈&#x1f923;&#x1f923;&#x1f923; Block comment &#xff1a; 多行注释 Doc comment&#xff1a;文档注释 Line comment&#xff1a;单行注释 小伙伴们可以改自己喜欢的颜色…

C++ String(2)

reserve 这个地方要和reverse区分清楚&#xff0c;reserve是保留的意思&#xff0c;而reverse是逆置的意思 reserve函数可以预先分配内存 reserve(n)代表至少保留可以容纳n个字符的空间&#xff08;具体多大和编译器有关&#xff09; 比如reserve(100)&#xff0c;代表开10…

网络基础Linux

目录 计算机网络背景 网络发展 认识 "协议" 网络协议初识 OSI七层模型 TCP/IP五层(或四层)模型 网络传输基本流程 网络传输流程图 ​编辑 数据包封装和分用 网络中的地址管理 认识IP地址 认识MAC地址 笔记&#xff08;画的图&#xff09; 协议&#x…

干货 | WiFi 7(802.11BE)技术规范详解

1 概述 1.1 简介 当前全球有近200亿的Wi-Fi设备正在使用&#xff0c;Wi-Fi已成为生活、工作中不可或缺的一部分。在实际应用中&#xff0c;Wi-Fi协议所传输无线流量&#xff0c;已占到无线总流量的90%。海量数据快速、安全传输受益于巨量Wi-Fi设备高效、安全、可靠地工作&a…

线程的状态

目录 一、线程的所有状态 二、状态转换 三、多线程初体验 一、线程的所有状态 状态是针对当前线程调度情况所描述的&#xff0c;又因为线程是调度的基本单位&#xff0c;所以我们所谈到的状态都是线程的属性。在java里对线程的状态&#xff0c;进行了一个更细的划分。 我们可…

基于SpringBoot网上超市的设计与实现录像

基于SpringBoot网上超市的设计与实现录像 SpringBoot网上超市的设计与实现录像

认证鉴权框架SpringSecurity-3--代码集成_认证篇1(重写UserDetailsService示例)

上一篇介绍了springSecurity中常用的组件和过滤器链&#xff0c;明白了springSecurity管理认证和授权的基本过程和所用到的组件。之后几篇我们通过在Java代码集成springSecurity&#xff0c;来学习下代码上认证是如何实现的。 代码上常用的认证方式有两种&#xff0c;一种是实现…

C语言操作符终极揭秘:表达式求值秘籍

C语言中操作符详解&#xff08;终结篇&#xff09; 放在最前面的表达式求值定义&#xff08;1&#xff09; 操作数的求值&#xff08;2&#xff09; 操作符的应用&#xff08;3&#xff09;类型转换&#xff08;3.1&#xff09;隐式类型转换&#xff08;Type Promotion&#xf…

[代码审计]宏*HCM最新文件上传漏洞分析复现

如果觉得该文章有帮助的&#xff0c;麻烦师傅们可以搜索下微信公众号&#xff1a;良月安全。点个关注&#xff0c;感谢师傅们的支持。 免责声明 本博客所发布的所有内容&#xff0c;包括但不限于信息、工具、项目以及文章&#xff0c;均旨在提供学习与研究之用。所有工具安全性…

谷歌Gemini发布iOS版App,live语音聊天免费用!

大家好&#xff0c;我是木易&#xff0c;一个持续关注AI领域的互联网技术产品经理&#xff0c;国内Top2本科&#xff0c;美国Top10 CS研究生&#xff0c;MBA。我坚信AI是普通人变强的“外挂”&#xff0c;专注于分享AI全维度知识&#xff0c;包括但不限于AI科普&#xff0c;AI工…

MYSQL SQL优化总结【快速理解】

1、优化insert操作 批量插入&#xff0c;防止大量与数据库进行访问 手动控制事务&#xff0c;减少事务的频繁开启和提交。 主键顺序插入 2、优化主键 主键优化的点就是避免主键过长&#xff0c;因为如果有二级索引&#xff0c;叶子节点存储的数据时间上是主键&#xff0c;如果主…