文档章节

一位 Rust 开发者的 Go 初体验

TiDB
 TiDB
发布于 03/05 21:05
字数 2873
阅读 4.1W
收藏 22

作者介绍:Nick Cameron,PingCAP 研发工程师,Rust 语言核心成员。

感谢 Rust 语言中文社区伙伴们的翻译和审校:

  • 翻译:尚卓燃
  • 审校:吴聪、张汉东

过去几周,我一直在用 Go 语言编写程序。这是我首次在大型且重要的项目中使用 Go。在研究 Rust 的特性时,我也看了很多关于 Go 的内容,包括体验示例和编写玩具程序。但真正用它编程又是一种完全不同的体验。 我觉得把这次体验写下来应该会很有趣。在这篇文章中,我会尽量避免将 Go 与 Rust 进行过多的比较,不过,由于我是从 Rust 转向 Go,难免也会包含一些比较。应该事先声明的是,我更偏袒 Rust ,但会尽力做到客观。

总体印象

用 Go 编程的感觉很棒。库程序里有我想要的一切,总体实现较为完善。学习体验也十分顺畅,不得不说,Go 是一种经过精心设计的实用性语言。举个例子:一旦你知悉了 Go 的语法,就能将其他语言中惯用法延续到 Go 中。只要你学会一些 Go,就可以相对轻易地推测 Go 语言的其他特性。凭借一些来自其他语言的知识,我能够阅读并理解 Go 代码,而不需要过多的搜索(Google)。

与 C/C++、Java、Python 等相比,Go 并没有那么多痛点,而且更具生产力。然而,它还是与这些语言处在同一个时代。尽管它从其他语言身上吸取了一些教训,甚至我个人认为它可能是那一代语言中最好的那个,但绝对还属于那一代语言。这是一种渐进式的改进,而不是推陈出新(需要明确的是,这不是意味着对其价值的批判,从软件工程的角度,渐进式改进通常会带来好的影响)。一个很好的例证是 nil:像 Rust 和 Swift 这样的语言已经去除了 null 的概念,并且消除了相关的一整类错误。Go 降低了一部分风险:没有空值(no null values),在 nil 0 之间进行区分。但其核心思想仍未改变,同样还会出现解空指针引用这种常见的运行时错误。

易学性

Go 非常易学。我知道人们经常吹捧这一点,但是我真的为自己生产力的飞速提高而感到震惊。多亏了 Go 语言以及它的文档和工具,我仅仅花了两天时间就可以写出「有价值」、可以提交的代码。

有助于易学性的几个因素是:

  • Go 很精简。很多语言都试图让自己看起来小巧,但 Go 真正做到了这一点(这基本上是一件好事,我对这种自律精神印象深刻)。

  • 标准库很出色(同样,也很小)。从生态系统中寻找并使用库程序非常容易。

  • 几乎没有其他语言中所不具备的东西。Go 从其他既存语言中提取了很多内容,并进行完善,最后将它们很好地组合在一起。它在避免标新立异这一方面做了极大努力。

乏味的样板式代码

Go 代码很快就会变得非常重复。这是由于它缺乏宏或者泛型这种用于减少重复的机制(接口虽然有利于抽象,但在减少代码重复方面作用没有那么大)。最终我会写很多函数,而他们除了类型不同之外其他甚至完全一样。

错误处理也会导致重复。许多函数中像 if err != nil { return err } 这样的样板式代码甚至比那些真正有价值的代码还要多。

使用泛型或宏来减少样板式代码有时会受到批评,理由是不应为使代码易于编写而使其丧失可读性。我发现 Go 恰恰提供了一个反例,复制和粘贴代码往往既快速又简单,阅读代码却会令人灰心丧气,因为你不得不忽略大量的无关代码或者在大量的相同代码中找到细微的不同。

我喜欢的东西

  • 编译时间:绝对快,可以确定要比 Rust 快得多。但实际上,它并没有我预期的那么快(对于中型到大型项目,我感觉它的速度只是与 C/C++ 相接近,或者稍微快一点。而我更加期待能够即时编译)。

  • 协程(goroutine)和信道(channel):值得称赞的是,Go 为生成协程和使用信道提供了轻量级的语法。尽管只是一个小细节,却使 Go 的并发编程体验比其他语言更优越,它真正揭示了语法的力量。

  • 接口:它们并不复杂,但是很容易理解和使用,并且在很多地方都很实用。

  • if ...; ... { } 语法:可以将变量的作用域限制在 if 语句真的很好。这与 Swift 及 Rust 中的 if let 起着相似的效果,但用途更为广泛(Go 没有像 Swift 和 Rust 那样的模式匹配,所以它无法使用 if let )。

  • 测试和文档注释都很容易使用。

  • Go 工具链非常友好:将所有东西都放在一个地方,而不需要在命令行上使用多个工具。

  • 拥有垃圾收集器(GC):不用考虑内存管理真的会使编程更加轻松。

  • 可变参数。

我不喜欢的东西

以下内容没有特定的顺序。

  • nil 切片:要知道 nilnil 切片和空切片三者都不相同,我敢保证我们只需要其中的两个,而不需要第三个。

  • 枚举类型并不是第一公民:使用常量模拟枚举让人感觉是一种倒退。

  • 不允许循环引用:这实际上限制了包在划分项目模块中的可用性,因为它变相鼓励了在一个包中堆积大量文件(或拥有大量零碎的小包,如果本该放在一起的文件四处分散,这也同样糟糕)。

  • switch 允许出现遗漏匹配的情况。

  • for ... range 语句会返回一对「索引/值」。要想只获取索引很容易(忽略值就好);但若要只获取值,则需要显式声明。在我看来,这种做法更应该颠倒过来,因为在大多数情况下,我更需要值而不是索引。

  • 语法:

    • 定义与用途存在不一致。
    • 编译器有时会很挑剔(例如,要求或禁止尾随逗号);通过良好的工具可以缓解这种困扰,但是有时仍然会产生一些恼人的额外步骤。
    • 使用多值返回类型时,类型上需要括号,但 return 语句中却不需要。
    • 声明一个结构体需要两个关键字(typestruct)。
    • 采用大写命名法来标记公共或私有变量,看起来就像匈牙利命名法那样,但更糟糕。
  • 隐式接口。我知道它也出现在我喜欢的东西中,但有时候它确实很惹人烦——特别是当你试图找出所有实现该接口的类型,或者哪些接口是为给定类型而实现的时候。

  • 你无法在不同的包中编写带有接收器的函数,所以即使接口是「鸭子类型」的,你也不能为其他包中的类型实现这个接口,这使得它们的用处大大降低。

  • 还有我之前已经提过的,Go 缺少泛型和宏。

一致性

作为一名语言设计者和程序员,Go 最让我惊讶的地方也许是它的内置功能和用户可用功能之间频频出现不一致。许多语言的目标之一就是尽可能消除编译器魔法,让用户也能使用内置功能。运算符重载是一个简单但有争议的例子。但 Go 有很多魔法!你很容易就会遇到这样的问题:无法做那些内置功能可以做的事情。

一些让我印象深刻的地方:

  • 返回多个值和信道的语法很棒,但是这两个无法一起使用,因为没有元组类型。

  • 能够用 for ... range 语句对数组和切片进行迭代,但对其他集合就无能为力了,因为它缺乏迭代器的概念。

  • len 或者 append 这样的函数是全局函数,但你自己的函数却无法转变成全局函数。这些全局函数只能使用内置类型。即便 Go「没有泛型」,它们也可以变得通用。

  • 没有运算符重载,那么 == 就会使人感到恼火。因为这意味着你不能在词典中使用自定义类型作为键,除非它们是可比较的。这一属性派生自类型结构,程序员无法重写该属性。

总结

Go 是一种简单、小巧、令人愉悦的语言。它也有一些犄角旮旯,但绝大部分是经过精心设计的。它的学习速度令人难以置信,并且规避了其他语言中一些不那么广为人知的特性。

Go 也是一种与 Rust 截然不同的语言。虽然两者都可以笼统地描述为「系统语言」或「C 语言的替代品」,但它们的设计目标、应用领域、语言风格和优先级不尽相同。垃圾收集确实带来了一个巨大的差异:使用 GC 使得 Go 变得更简单、更小,也更容易理解。而不使用 GC 使 Rust 奇快无比(特别是在您需要保证延迟,而不仅仅是高吞吐量的时候),并且得以支持 Go 中不可能实现的特性或编程模式(或者至少在不牺牲性能的前提下是无法实现的)。

Go 是一种编译型语言,其运行时得到了良好的实现,其速度毋庸置疑。Rust 也是编译型语言,但是运行时要小得多,它真的迅捷无比。在没有其他限制的情况下,我认为选择使用 Go 还是 Rust 其实意味着一种权衡:

  • 一方面,Go 的学习曲线更短、程序更简单(这意味着更快的开发速度);

  • 另一方面,Rust 真的性能卓越,并且类型系统更富有表现力(这使程序更安全,也意味着更快的调试和错误查找)。

附:英文原版文章

欢迎大家在下方留言,说说你的 Go & Rust 体验吧:从入门到放弃,从放弃到精通,踩过的坑,流过的泪……或者说说它们的必杀优势。

⚠️理智探讨,Peace & Love.

© 著作权归作者所有

TiDB

TiDB

粉丝 249
博文 329
码字总数 896396
作品 4
海淀
私信 提问
加载中

评论(24)

TavenLi
TavenLi
我觉得,应该拿Go 与 Java 、PHP 、C# 比较,Rust 与 C/C++ 比较,因为Go本身就与 Rust 是有根本上的区别的,拿Go与Rust比,不太妥。
行者__
行者__
用了 clojure 你就拥有了 go c# java 等语言的免疫力了. 随便怎么吹,可以做到我自巍然不动 :)
exten
exten
Go包管理不清晰,主要在,方法、函数、结构、包、文件,这之间如何编排的问题。乏味的样板式代码,这个说的非常正确,基本每个API调用都要判断 err==nil,占用了大量逻辑代码空间。以及上文中所说的nil、nil切片、空切片,我只想要nil和空切片。有些地方go还是有待商榷的。我期望go能优化错误捕获、范型、宏定义。
漫步海边小路
漫步海边小路
golang学起来简单,但缺点也不少,总体来说,写起来和python的效率差不多,可用的包都比较多,生态比较稳定.rust学习曲线略陡峭.

人生苦短,何不找个不太费心力的语言?
StevenDisonTangor
StevenDisonTangor
不知道为什么rust,没有i++...
Alex-loongkylin
Alex-loongkylin
如果你觉得go好,那你一定要试试dlang,你会觉得超级好。
李嘉图
李嘉图
乏味的样板式代码 都2020了,还有语言要靠这种东西,我是不喜欢
晒太阳的小猪
晒太阳的小猪
看来都得学习了解一下
LeeNux
LeeNux
范型范型
cardinalinux
cardinalinux
不允许循环引用这点真是太蛋疼了。
helloclia
helloclia
强制你使用接口和依赖注入😅
rust语言初体验

Rust介绍: Rust 是一门系统级编程语言,被设计为保证内存和线程安全,并防止段错误。作为系统级编程语言,它的基本理念是 “零开销抽象”。理论上来说,它的速度与 C / C++ 同级。Rust 可以...

有力量的神经病
2016/08/12
789
0
2018 年开源技术 10 大发展趋势

你是否关注过开源技术的发展趋势? 这里是 10 个预测。 技术一直在变革,诸如 OpenStack、增强型网页应用Progressive Web App(PWA)、Rust、R、认知云the cognitive cloud、人工智能(AI),...

作者: Sreejith Omanakuttan
2017/12/27
0
0
Rust 2017 调查报告:学习曲线是最大痛点

Rust 官方在社区上做了一次调查,以了解用户如何看待 Rust 的发展。调查共收到 5368 份回复,其中有 大约 2/3 的是 Rust 用户,剩下的 1/3 是非 Rust 用户,调查结果如下。 点此查看完整调查...

王练
2017/09/07
6.5K
53
前端每周清单半年盘点之 WebAssembly 篇

前端每周清单专注前端领域内容,以对外文资料的搜集为主,帮助开发者了解一周前端热点;分为新闻热点、开发教程、工程实践、深度阅读、开源项目、巅峰人生等栏目。欢迎关注【前端之巅】微信公...

王下邀月熊
2017/09/14
0
0
Rust 2018 年度调查报告发布

Rust 官方博客发布了 2018 年的 Rust 调查报告。 今年是第三次对 Rust 进行年度调查,首次增加了面向英语以外语言环境的调查,在全球 14 种语言背景的调查下,总共收集到了 5991 份调查记录,...

h4cd
2018/11/29
3.2K
7

没有更多内容

加载失败,请刷新页面

加载更多

北京哪里有开住宿费发票

电13564998196 陈生幵-真-嘌,百-分-百-真。-本报讯(劳动报记者陆燕婷)聚焦餐饮行业,58同城招聘研究院昨 发布数据显示,今年上半年,全国餐饮行业招聘需求增长46.18%,平均月薪6387元。随着...

票迦徴fp2090
10分钟前
11
0
北京哪里有开住宿费发票

电13564998196 陈生幵-真-嘌,百-分-百-真。-本报讯(劳动报记者陆燕婷)聚焦餐饮行业,58同城招聘研究院昨 发布数据显示,今年上半年,全国餐饮行业招聘需求增长46.18%,平均月薪6387元。随着...

嘌迦徴fp2090
11分钟前
13
0
北京哪里有开住宿费发票

电13564998196 陈生幵-真-嘌,百-分-百-真。-本报讯(劳动报记者陆燕婷)聚焦餐饮行业,58同城招聘研究院昨 发布数据显示,今年上半年,全国餐饮行业招聘需求增长46.18%,平均月薪6387元。随着...

嘌珈徴fp2090
11分钟前
11
0
北京哪里有开住宿费发票

电13564998196 陈生幵-真-嘌,百-分-百-真。-本报讯(劳动报记者陆燕婷)聚焦餐饮行业,58同城招聘研究院昨 发布数据显示,今年上半年,全国餐饮行业招聘需求增长46.18%,平均月薪6387元。随着...

票加徴fp2090
12分钟前
5
0
北京哪里有开住宿费发票

电13564998196 陈生幵-真-嘌-本报讯(劳动报记者陆燕婷)聚焦餐饮行业,58同城招聘研究院昨 发布数据显示,今年上半年,全国餐饮行业招聘需求增长46.18%,平均月薪6387元。随着餐饮行业的快速发...

嘌加徴fp2090
13分钟前
7
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部