对比着 Npm 来学 Rust 的 Cargo,一遍就会了

4,322 阅读5分钟

我们写 Node.js 代码离不开的是什么,是 Npm。这就像写 Rust 代码离不开 Cargo 一样。

大家可能不了解 Rust 和 Cargo。没关系,接下来我们就对照着 npm 来学习下 Cargo,帮大家入门下 Rust。

我们先从熟悉的 Node.js 开始:

写 Node.js 代码的流程

我们会先创建一个目录,然后在目录下执行 npm init

mkdir node-pro
cd node-pro
npm init

控制台会交互式的询问一些信息,然后生成 package.json

也可以加上 -y 用默认值快速创建。

之后我们会用 npm install xxx 来安装依赖,比如安装 babel

npm install babel

这时候 package.json 的 dependencies 下就有了 babel 的依赖信息

同时会生成一个 package-lock.json 记录依赖版本(用 yarn 安装的话就是 yarn.lock 文件)

依赖会安装在 node_modules 下。

之后呢,我们会创建 src,写一些代码。

然后在 package.json 的 scripts 中指定编译、测试等脚本:

然后代码写完以后,通过 npm run build 来跑构建、通过 npm run test 来跑测试:

npm run build
npm run test

生成的产物可以通过 npm publish 发布到 npm 仓库。

过程中可能还用到一些其他的命令: 比如要更新依赖会用 npm update,比如要搜索依赖用 npm search 等。

这就是一个 Node.js 项目的开发流程,抛开具体写的 Node.js 代码不谈,整个工具链是用 npm 串联的。

你可能会说这个很基础啊,不是入门的内容么?

是的,npm 是 Node.js 入门要学的,那么同样,入门 Rust 也要先学 Cargo。

而且,npm 和 cargo 的设计特别的像,几乎看一遍大家就会了。

那还等什么,赶紧来学一下 Cargo 吧。

写 Rust 代码的流程

和 Node.js 项目一样,可以先创建目录,然后使用 cargo init 初始化代码(这里也可以直接使用 cargo new,效果和下面 mkdir + cargo init 一样)

mkdir rust-pro
cd rust-pro
cargo init

然后就会创建这样的目录结构:

cargo.toml 相当于 package.json,也是声明包信息和 dependencies 的。

而且连 src 都有了, git 也初始化了,这比 npm init 更贴心啊(难怪大家都喜欢 rust,这工具链做的多细)。

cargo.toml 的内容是这样的:

[package] 下声明的是包的信息,[dependencies] 下声明的是依赖信息。

我们用 cargo search 搜索下某个包(相当于 npm search):

可以搜到 html2md 的版本是 0.2.13,我们把它填到依赖里:

之后我们写点代码,把 html 转成 markdown:

然后,编译和执行:

cargo build
cargo run

就可以看到执行结果:

我们用 npm run build 执行的也是构建命令,只不过是自己配置的三方编译工具,而 cargo 是用内置的编译工具。

这样我们就跑起来了第一个 rust 程序。是不是流程和 npm 有那么一丢丢的像。

而且,像 yarn.lock 或者 package-lock.json 一样,cargo 也有 Cargo.lock 来记录了依赖的具体信息:

后续也可以执行 cargo test 来跑测试代码,可以执行 cargo publish 来上传到中央仓库。和 npm 的整体流程比较类似。

Npm 和 Cargo 相似的原因

为什么 cargo 和 npm 这么类似呢?

这说明这已经是最佳实践了!也就是把 init 的脚手架、编译构建、运行、测试、发布等功能集成到一个命令中工具中,内置到语言的工具链。

对照下古老的 C++ 就能看出区别:

C++ 使用 clang 或者 gcc 编译,其他的功能并没有,需要结合 cmake 来声明一些其他的命令。最关键的是没有中央的仓库和依赖管理工具,每个依赖都要手动下载,然后放到项目目录下,特别麻烦。

既然这是必备功能,为什么不内置到语言的工具链呢?

所以 npm 和 cargo 都把 init、install、update、build、test、publish 等命令内置了,而且也都支持了中央仓库和依赖管理。

这是现代的语言工具链的最佳实践了,用别的现代语言的工具也会感觉差不多。

总结

Cargo 之于 Rust 就像 Npm 之于 Node.js,两者都是初始化、依赖管理、构建、发布、等的集成的命令行工具链。

Node.js 的项目的开发流程是这样的:

  • npm init 初始化项目
  • npm search 搜索依赖
  • npm install 安装依赖
  • npm update 升级依赖
  • npm run build 执行构建
  • npm run test 执行测试
  • npm publish 发布到中央仓库

Rust 项目的开发流程也类似:

  • cargo init 初始化项目 (或者 cargo new,这个相当于 mkdir + cargo init)
  • cargo search 搜索依赖
  • cargo install 安装依赖
  • cargo update 升级依赖
  • 手动把依赖填到 Cargo.toml 中
  • cargo build 编译构建代码
  • cargo run 运行代码
  • cargo test 跑单元测试
  • cargo publish 发布到中央仓库

虽然具体的语法不同,项目结构也有差别,但是整个工具链的流程是类似的。这是现代语言工具链的最佳实践了。

相比之下,C++ 没有依赖管理,没有集成的工具链,开发体验远远比不上有 Cargo 的 rust 和有 npm 的 Node.js。

其实我们学习 rust 或其他语言,都可以对比我们熟悉的 JS 来学,因为他们只是使用计算机的不同的抽象,面对的问题差不多,只不过解法不同,对比着学习,效率会更高。

对比着 Npm 来学 Cargo,是不是看一遍就会了~