A developer of Rust Go early experience

Author | Nick Cameron

Zebian | Guo Rui

Over the past few weeks, I have been programming with Go language. This is my first time Go large and important project. In studying the characteristics of Rust, I also saw a lot of content on Go, including examples and experience to write toy programs. But the real programming with it is a completely different experience.

I think to write down this experience should be very interesting. In this article, I will try to avoid undue Go compared with Rust, though, because I was turned to Go from Rust, inevitably also contain some more. It should be declared in advance that I'm more partial to Rust, but will try to be objective.

Overall impression

Go programmed with a great feeling. Library program has everything I want, overall achieve a more perfect. Learning experience is also very smooth, have to say, Go is a carefully designed practical language. For example: Once you are aware of the Go's syntax can be extended to other languages, idioms Go in. As long as you learn some Go, it can relatively easily guess other features of the Go language. With some knowledge from other languages, I was able to read and understand the code Go, without undue search (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 语句中却不需要。

  • 声明一个结构体需要两个关键字(type 和 struct)。

  • 用大写命名法来标记公共或私有变量,看起来就像匈牙利命名法那样,但更糟糕。

隐式接口。我知道它也出现在我喜欢的东西中,但有时候它确实很惹人烦——特别是当你试图找出所有实现该接口的类型,或者哪些接口是为给定类型而实现的时候。

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

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

一致性

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

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

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

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

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

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

总结

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

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

Go 是一种编译型语言,其运行时得到了良好的实现,其速度毋庸置疑。Rust 也是编译型语言,但是运行时要小得多,它真的迅捷无比。在没有其他限制的情况下,我认为选择使用 Go 还是 Rust 其实意味着一种权衡:一方面,Go 的学习曲线更短、程序更简单(这意味着更快的开发速度);另一方面,Rust 真的性能卓越,并且类型系统更富有表现力(这使程序更安全,也意味着更快的调试和错误查找)。

声明:本文来源于PingCAP,系作者投稿,版权归其所有。

推荐阅读 

腾讯云“抢救”微盟!开 766 次在线会议、调拨 100 多台服务器、闹钟只敢定 2 小时

重磅!教育部再次审批 179 所高校新增本科 AI 专业

人工智能改变未来教育的5大方式!

Linux 会成为主流桌面操作系统吗?

6 个步骤,教你在Ubuntu虚拟机环境下,用Docker自带的DNS配置Hadoop | 附代码

开发项目时如何选择区块链平台?我们分析了以太坊、Bitcoin via RSK、Ardor三个有趣的平台来给你回答!

你点的每一个在看,我认真当成了喜欢

发布了1792 篇原创文章 · 获赞 4万+ · 访问量 1618万+

Guess you like

Origin blog.csdn.net/csdnnews/article/details/104708120