全文:Python 和 Rust,都是近些年颇受合作开发人员钟爱的C语言,所以做为两个保有二十年 Python 程式设计实战经验的合作开发人员而言,首度试著 Rust 会有什么样的体会呢?
镜像:https://karimjedda.com/carefully-exploring-rust/
译者 | Karim Jedda
翻译者 | 弯月 白眉林 | 郑丽媛
公司出品 | CSDN(ID:CSDNnews)
前段时间,我找出了这份新组织工作,公司最常见的C语言众所周知是 Rust。
在此之后,我采用 Python 长达二十年之久,主要是做统计数据工程建设的组织工作。但现如今,我急于试著呵呵此种捷伊(对我而言)C语言。我时常在相同网络平台上看见各种类型夸奖 Rust 的该文,我想看一看 Rust 与否吗无愧盛誉。
Rust 与 Python 有十分大的相同,因而我不急于在责任编辑中详尽表明 Rust 的独有之处。做为新手,我只期望尽早上手,期望更有吸引力最短的过渡阶段,尽早用 Rust 顺利完成组织工作,与此同时也期望评估结果呵呵我自己的自学潜能。
从某种意义上而言,我更钟爱的是 Rust 总体的采用新体验,而不是具体内容的机能条目。
增设合作开发自然环境
增设合作开发自然环境比较简单,只需参考 Rust 中文网站提供更多的实例,在终端产品中运转两个指示就可以了。
当你认为所有人都已加装并实用性恰当,这时如果想校正 Rust 与否已恰当加装,只需在空目录中建立两个空工程项目:
cargo new tutorial
cd tutorial
cargo run
接下来,在文本编辑器中打开该文件夹。如果你是新手,我推荐 VSCode,因为其中的一些扩展很有帮助,关于如何采用这些扩展的指南也很容易入手。我推荐 rust-analyzer 做为 VSCode 的唯一扩展。
输出与调试
如果想了解程序是如何运转的,首先要做的就是通过指示行来了解程序在干什么,以及顺利完成了什么。
此外,你还可以采用常规调试器。在 M1 上,我推荐在 Visual Studio Code 中采用 LLDB,它不仅组织工作良好,通常还要比在输出结果中打印日志更为方便。
到这里为止,Rust 与 Python 其实都十分相似,只不过所有指示都是通过 cargo run 运行的,而不是调用特定文件,如 python3 somefile.py。
另外,你也可以先运转 cargo build,然后运转 target/debug/tutorial 中的文件,得到的结果是相同的。接下来,如果将生成的文件复制到另两个位置或另一台类似的机器上,也可以正常组织工作,且无需加装任何与 Rust 相关的东西。
错误处理
不得不承认,程式设计中总会遇到一些意外,能够以可预测的方式处理这些意外十分重要。程式设计中的一大挑战就是很难考虑到程序中所有可能出现的错误,因为只要写代码就可能会出错。
“每个人都知道调试比编写程序要难一倍。所以,如果你在代码编写代码时就用尽了聪明才智,又如何调试呢?”
—— Brian W. Kernighan
在 Python 中,通常我们通过 try/except 方法来抛异常,并顺利完成错误处理。我们运转一段代码,如果出错,则通过条件来捕捉异常,如果所有条件都不匹配,则将其放入一个通用的异常中。异常有各种相同的类别,Python 允许你在包中调用不存在的函数,并在运转时产生异常,但在 Rust 中这样做甚至无法通过编译——Rust 不允许在运转时出现任何奇怪的错误,从而消除了一大类不太容易预测的错误。
下面通过两个例子而言明 Rust 的此种方法,与此同时我会用 Python 的术语进行解释。
在上面的代码中,我们建立了两个自定义的异常,在 do_something 函数中抛出,而 main 函数会检查该异常。上面的代码跟 try/exept 基本上一样,只不过多了一些样板代码(这些样板代码是必须的,但以我现在的水平有点难以理解)。
你也许会说“肯定有更好的办法”,特别是如果你有很多 Python 实战经验的话,的确如此,我们将不得不采用包。。
采用外部包
与其他行业相比,程式设计的最大优势就是可以采用别人构建的东西。如果你计划在程序中进行错误处理,所以有两个很好用的包 thiserror。Rust 的包管理器是 cargo。
Rust 中的包叫做 crate。加装方法为编辑目录下的 Cargo.toml 文件。在本例中,我们在 [dependencies] 后面添加 thiserror = “1.0” 就可以了。
然后可以像下面这样重写之前的代码:
现在代码看起来很正常。与所有人从头开始相比,我更喜欢此种做法。
我花了四五年时间才找出用 Python 程式设计的乐趣,所以我也愿意多花些时间来探索 Rust 的高级机能。Rust 有许多错误处理的方法,而我喜欢更简单的方法。
我有意略过了一些简单的概念,比如“什么是 enum?” “pub 是什么意思?” “那些#标记是什么”等,因为你只需运转呵呵代码就能明白它们的意思。
所有人看起来都还不错。所以,测试方面又如何呢?
编写测试
测试应该从单元测试和集成测试两个级别上着手。实现方法有好几种,虽然你可以把测试代码和 Rust 代码放在同两个文件中(这也是官方指南的推荐),但我还是期望用两个单独的文件夹来组织所有测试代码,这样可以减少阅读代码时的负担,也可以减少编辑文件时占用的屏幕面积。而且说实话,在编写测试和编写代码时,我的心情是不一样的。
方法众所周知如下:
然后可以用 cargo test 运转测试,结果如下:
还有许多值得展开讲的地方,但为了避免过于复杂,我们点到为止,这算是“帕累托最优”(又称80/20法则)。这让我想起了 pytest,一个能即刻提高舒适度的工具。
读取文件、运转一些代码并写入另两个文件
以上,我们讨论了一些最基本的问题:输出,调试,采用外部包,以及测试。下面,我们来做一些更有效率的事情:我们可以写两个程序来处理本地文件。下面的例子将会读取 CSV 文件,计算一些数值,然后将输出结果。
为了实现该程序,我们需要在 Cargo.toml 中添加以下两行增设:
csv=”1.1″
serde={version=”1″, features=[“derive”]}
你可以猜猜 main() 函数应该什么样写。
当然,这个程序还可以实现更多机能。如果你有两个十分复杂的 CSV 文件,则可以在 Rust 中调用 pandsa(pola.rs)来处理统计数据。我还需要进一步研究,不过似乎此种处理方法十分强大且高效。
我认为,与 Python 相比,Rust的 CSV 处理潜能不相上下,除了它能自动反序列化之外。
最后,我们还可以添加一些测试,此处不再赘述。
发送 HTTP 请求
下面,我们来试著发送基本的 HTTP 请求并处理结果。现在的绝大多数请求都需要处理 JSON。
在 Cargo.toml 中添加如下几行代码:
reqwest = { version = “0.11”, features = [“json”] }
tokio = { version = “1”, features = [“full”]}
serde_json = “1”
这样这一点吗?在这方面 Rust 做得很好。
我相信,Rust 的生态系统会越来越大,以覆盖更多的用例,以后利用已有的 crate 实现这所有人会易如反掌。
采用 SQLite
虽然在这篇该文中提到 SQLite 似乎有些奇怪,但我合作开发过的程序时常见到 SQLite。我很喜欢 SQLite,因为它是可移植的,十分有效,而且不需要任何维护。
用 Python 操作 SQLite 的问题永远是要不要采用 ORM。请不要误会,SQLAlchemy 十分杰出,但在进行十分小的操作时,用它就像杀鸡用牛刀了。而且 SQLAlchemy 带来的复杂性使它不适合小型嵌入式设备。
反之, Rust 可以在这方面大放异彩,网上有很多如何利用Rust操作 SQLite 的例子,我认为都十分不错。
举个简单的例子,别忘了在 Cargo.toml 中添加下面这行代码:
rusqlite = { version = “0.28.0”, features = [“bundled”] }
该例子来自 rusqlite crate。当然,这只是冰山一角。但组合以上几种方法,就可以实现许多很有用的机能了。
总结
综合考虑,Rust 是两个十分杰出的语言,有许多杰出的包,十分感谢发明这门语言并为之努力贡献的合作开发人员们。虽然这篇该文只是对 Rust 做了初步的探索,但我期望抛砖引玉,让新手产生自学 Rust 的兴趣。
首度采用某种C语言时,重点在于弄清楚语言本身能实现哪些机能,而不是背诵一篇完整的术语表。你不需要去理解 borrowing、继承或 traits 的具体内容含义,而应该跟随些入门该文按部就班地做一遍。
从无到有的难度远大于从一到十。