go語言與rust,go語言與java的區(qū)別

我為什么要選擇Rust

你好,很高興為你解答。

專注于為中小企業(yè)提供成都做網(wǎng)站、網(wǎng)站制作、成都外貿(mào)網(wǎng)站建設(shè)服務(wù),電腦端+手機(jī)端+微信端的三站合一,更高效的管理,為中小企業(yè)溫嶺免費做網(wǎng)站提供優(yōu)質(zhì)的服務(wù)。我們立足成都,凝聚了一批互聯(lián)網(wǎng)行業(yè)人才,有力地推動了數(shù)千家企業(yè)的穩(wěn)健成長,幫助中小企業(yè)通過網(wǎng)站建設(shè)實現(xiàn)規(guī)模擴(kuò)充和轉(zhuǎn)變。

專訪資深程序員莊曉立:我為什么要選擇Rust?

Rust是由Mozilla開發(fā)的注重安全、性能和并發(fā)性的編程語言。這門語言自推出以來就得到了國內(nèi)外程序員的大力推崇。Rust聲稱解決了傳統(tǒng)C語言和C++語言幾十年來飽受責(zé)難的內(nèi)存安全問題,同時還保持了極高的運行效率、極深的底層控制、極廣的應(yīng)用范圍。但在國內(nèi)有關(guān)Rust的學(xué)習(xí)文檔并不多見,不久前,筆者聯(lián)系上了Rust1.0版本代碼貢獻(xiàn)者莊曉立(精彩博文:為什么我說Rust是靠譜的編程語言),請他分享Rust語言特性以及學(xué)習(xí)經(jīng)驗。

CSDN:你是從什么時候開始接觸Rust語言的?是什么地方吸引了你?

莊曉立:我大概從2013年后半年開始深入接觸Rust語言。它居然聲稱解決了傳統(tǒng)C語言和C++語言幾十年來飽受責(zé)難的內(nèi)存安全問題,同時還保持了極高的運行效率、極深的底層控制、極廣的應(yīng)用范圍。

其ownership機(jī)制令人眼前一亮,無虛擬機(jī)(VM)、無垃圾收集器(GC)、無運行時(Runtime)、無空指針/野指針/內(nèi)存越界/緩沖區(qū)溢出/段錯誤、無數(shù)據(jù)競爭(Data Race)……所有這些,都深深地吸引了我——這個十多年以來深受C語言折磨的痛并快樂著的程序員。

CSDN:在你看來,Rust是怎樣的一門語言?它適合開發(fā)什么類型的項目?為何你會說Rust不懼怕任何競爭對手,它既能取代C語言地位;又可挑戰(zhàn)C++市場,還可向Java、Python分一杯羹?與這些語言相比,Rust有哪些優(yōu)越的特性?

莊曉立:Rust是一門系統(tǒng)編程語言,特別適合開發(fā)對CPU和內(nèi)存占用十分敏感的系統(tǒng)軟件,例如虛擬機(jī)(VM)、容器(Container)、數(shù)據(jù)庫/游戲/網(wǎng)絡(luò)服務(wù)器、瀏覽器引擎、模擬器等,而這些向來主要都是C/C++的傳統(tǒng)領(lǐng)地。

此外,Rust在系統(tǒng)底層開發(fā)領(lǐng)域,如裸金屬(bare metal)、操作系統(tǒng)(OS)、內(nèi)核(kernel)、內(nèi)核模塊(mod)等,也有強(qiáng)勁的實力,足以挑戰(zhàn)此領(lǐng)域的傳統(tǒng)老大C語言。Rust豐富的語言特性、先進(jìn)的設(shè)計理念、便捷的項目管理,令它在上層應(yīng)用開發(fā)中也能大展拳腳,至少在運行性能上比帶VM和GC的語言要更勝一籌。無GC實現(xiàn)內(nèi)存安全機(jī)制、無數(shù)據(jù)競爭的并發(fā)機(jī)制、無運行時開銷的抽象機(jī)制,是Rust獨特的優(yōu)越特性。

其他語言很難同時實現(xiàn)這些目標(biāo),例如傳統(tǒng)C/C++無法保證內(nèi)存安全,Java/Python等無法消除運行時開銷。但Rust畢竟還是很年輕的項目,它釋放影響力需要時間,被世人廣泛接受需要時間;它的潛力能否爆發(fā)出來,需要時間去檢驗。我們只需耐心等待。

CSDN:Rust在國內(nèi)有沒有具體的實際使用案例?

莊曉立:因為Rust1.0正式版剛剛發(fā)布不足一月,在國內(nèi)影響力還不大,我們不能苛求它在國內(nèi)有實際應(yīng)用案例。但是在國外,一兩年前就已經(jīng)有OpenDNS和Skylight把Rust應(yīng)用在生產(chǎn)環(huán)境。還有瀏覽器引擎Servo、Rust編譯器和標(biāo)準(zhǔn)庫、項目管理器Cargo等“兩個半大型應(yīng)用案例”。這些足夠說明Rust語言的成熟和實用。

CSDN:你參與了Rust1.0版本代碼貢獻(xiàn),目前該版本正式版已經(jīng)發(fā)布,對此你感覺如何?這門語言是否已經(jīng)達(dá)到比較成熟的階段?

莊曉立:我積極參與了Rust語言開源項目,多次貢獻(xiàn)源代碼,曾連續(xù)三次出現(xiàn)在Rust官方博客公布的Rust 1.0 alpha、Rust 1.0 beta和Rust 1.0正式版的貢獻(xiàn)者名單中。在Rust 1.0正式版出臺的過程中及此前的很長一段時間,開發(fā)者付出了極大的努力,確保Rust 1.0正式版在Semver 2.0規(guī)范下,務(wù)必保持向后兼容性,除非遇到重大Bug不得不修復(fù)。

我認(rèn)為,在1.0正式發(fā)布之后,Rust就已經(jīng)進(jìn)入了比較成熟的階段。而且,Rust還在快速迭代發(fā)展過程中,1.0發(fā)布6周后將發(fā)布1.1,再6周后將發(fā)布1.2,必然會一步一個臺階,越來越成熟穩(wěn)定。

CSDN:除了功能優(yōu)先級以外,在你看來,Rust正在朝什么方向發(fā)展?未來的Rust可以期待什么樣的特性?

莊曉立:Rust一定會沿著“確保內(nèi)存安全、無運行開銷、高效實用”的既定方向持續(xù)發(fā)展。在短期內(nèi)值得期待的語言特性有:動態(tài)Drop、偏特化、繼承、改進(jìn)borrow checker、改進(jìn)宏和語法擴(kuò)展。短期內(nèi)值得期待的其他特性有:增強(qiáng)文件系統(tǒng)API、提供內(nèi)存申請釋放API、更好地支持Windows和ARM、更快的編譯速度、更方便的二進(jìn)制分發(fā)機(jī)制(MUSL)、更實用的工具等等。

CSDN:據(jù)我了解,你之前也比較推崇Go語言,為何想到放棄Go轉(zhuǎn)向Rust?

莊曉立:推崇Go語言還談不上,不過我曾經(jīng)嘗試努力接受Go語言,2011底年開始我曾經(jīng)花費將近半年時間深度關(guān)注Go開發(fā)進(jìn)程,提了很多具體的改進(jìn)意見和建議,也曾經(jīng)多次嘗試貢獻(xiàn)源代碼。后來考慮到Go語言的設(shè)計理念跟我偏差太大,其社區(qū)也不太友好,慢慢地疏遠(yuǎn)了它。我曾經(jīng)寫過一篇博客《我為什么放棄Go語言》,談到了很多具體的原因。

CSDN:國內(nèi),參與Rust代碼貢獻(xiàn)的開發(fā)者多嗎?有核心的人員嗎?有哪些社區(qū)在維護(hù)Rust?

莊曉立:國內(nèi)參與Rust代碼貢獻(xiàn)的開發(fā)者并不多,但也不少,官方的貢獻(xiàn)者名單中也偶見幾個貌似國人的名字。Rust的核心開發(fā)人員基本上都是Mozilla公司的員工,他們專職負(fù)責(zé)開發(fā)維護(hù)Rust語言和相關(guān)的項目,Rust社區(qū)也主要是他們參與組織和管理的。社區(qū)人員討論主要集中在GitHub項目主頁RFC/PR/Issue官方、Discuss論壇/IRC、Reddit、HN、StackOverflow等。

每個開發(fā)人員都應(yīng)該知道的16個頂級新計算機(jī)編程語言

函數(shù)式語言

Elixir

Elixir 比 Erlang 更容易編寫,具有 Haskell 等語言的函數(shù)式編程概念。Elixir是基于Erlang 虛擬機(jī)的,其廣為人知的特點是運行低延時、分布式、可容錯的系統(tǒng),并成功用于Web開發(fā)與嵌入式軟件領(lǐng)域。

Elm

Elm是一種用于構(gòu)建 Web 應(yīng)用程序的函數(shù)式語言。業(yè)內(nèi)一般認(rèn)為,它適用于創(chuàng)建高可交互應(yīng)用,例如復(fù)雜的用戶界面,開發(fā)人員可以通過 Elm 快速編寫富有表現(xiàn)力的系統(tǒng)。Elm 也以沒有運行時異常而聞名。

PureScript

PureScript是一種可編譯為 JavaScript 的純函數(shù)式編程語言。與 Haskell 最相似的是,PureScript 最適合用于開發(fā) Web 應(yīng)用程序和服務(wù)器端應(yīng)用程序。

PureScript 支持類型推斷,與其他語言相比,需要明顯類型注釋要少得多。

Swift

Swift是一種由蘋果公司開發(fā)的通用編譯編程語言,最早的設(shè)想是替代上一代編程語言O(shè)bjective-C ,過程中結(jié)合了Objective-C、Rust、Ruby 和 Python等語言的編程思想。目前Swift用于開發(fā)蘋果自己的手機(jī)、服務(wù)器、臺式機(jī)上的應(yīng)用軟件。

程序語言

Go

Go語言是由谷歌公司創(chuàng)造的類似C風(fēng)格的語言。Go 比 C++ 或 Java 更簡潔,比 Ruby 或 Python 更安全。

一些缺點: 編碼要求嚴(yán)格。比如,不能混用符號和無符號整數(shù)。還有一個明顯的遺漏,Go語言沒有泛型和繼承。

但Go語言的優(yōu)勢同樣明顯,簡單且易于使用。Go語言擅長于網(wǎng)絡(luò)和多線程方面的編程。

面向?qū)ο笳Z言

DART

Dart同樣來自谷歌公司具有C語言風(fēng)格。Dart可以輕松編寫JavaScript、Java for Android、本地機(jī)器代碼或獨立的 Dart 虛擬機(jī)。它還可以運行后端代碼。

Dart 非常適合使用事件驅(qū)動代碼構(gòu)建用戶界面。根據(jù)Dart 團(tuán)隊成員的說法,Dart的優(yōu)勢:可選的靜態(tài)類型、最小的編譯時錯誤和強(qiáng)大的內(nèi)置編輯器。

Pony

Pony是一種基于無數(shù)據(jù)競爭類型和垃圾收集的語言,并使用 actor 模型以及稱為引用功能的東西。

你可以把 Pony 想象成某種“Rust 遇上 Erlang”的復(fù)合體,沒有鎖,高并發(fā)是其主要優(yōu)點。

Pony 的缺點是 API 穩(wěn)定性低、很少有高質(zhì)量的第三方庫和有限的本地工具。

TypeScript

TypeScript是一個基于 JavaScript 靜態(tài)類型定義構(gòu)建,并由微軟維護(hù)且開源編程語言。Visual Studio Code 或Visual Studio 是推薦的IDE編輯器,微軟大廠的用戶體驗和錯誤檢查也不用懷疑。

復(fù)合編程語言

Hack

Hack是一種作為 PHP 方言的 HipHop 虛擬機(jī)的編程語言。于 2014 年由Facebook創(chuàng)建,允許程序員同時使用靜態(tài)和動態(tài)類型(也稱為漸進(jìn)類型),這為編碼提供了靈活性。

Julia

Julia是一種高級通用編程語言,用于計算科學(xué)和數(shù)值分析。Julia 以動態(tài)類型和可重現(xiàn)的高性能特性而聞名。

Julia 在數(shù)據(jù)可視化和機(jī)器學(xué)習(xí)等方面都有大量用途。事實上,它被英國保險公司 Aviva 用于風(fēng)險計算,紐約聯(lián)邦儲備銀行用于金融建模,甚至氣候建模聯(lián)盟用于氣候變化建模。它擁有Fortran、C++、R、Java、C 、Python等的接口,這使其成為最受追捧的新語言之一。

Kotlin

Kotlin是運行在 Java 虛擬機(jī)中的更快、更流暢的 Java 版本。它現(xiàn)在是Android 開發(fā)的首選語言。根據(jù) Android 開發(fā)者網(wǎng)站顯示,程序員正轉(zhuǎn)而采用 Kotlin,因為該語言的樣板代碼更少,空指針異常更少,并且與 Java 有互操作性。

Kotlin 可用于在 iOS 和 Android 上運行的應(yīng)用程序、不使用額外運行時或虛擬機(jī)。

Nim

Nim是一種優(yōu)先考慮可讀性的靜態(tài)類型語言。通過結(jié)合多種語言的特性,Nim 為程序員提供了速度和易用性。

它帶有 JavaScript 后端、分散的包管理、自動內(nèi)存管理、C 和 C++ 庫的綁定以及用于調(diào)試的回溯。作為一種語言,Nim 是有限的,但它包含一組元編程功能,如泛型、模板和宏,因此開發(fā)人員可以在避免冗長代碼的同時以不同的風(fēng)格工作。

OCaml作為此列表中較舊的語言,OCaml是一種多范式語言——既有函數(shù)式、命令式和類型安全,也具有面向?qū)ο蠊δ堋?/p>

OCaml 的一些優(yōu)勢:定義數(shù)據(jù)類型很容易。默認(rèn)情況下,所有變量都是不可變的。API 穩(wěn)定,具有良好的庫向后兼容性。該語言還為獨立應(yīng)用程序提供自動內(nèi)存管理和單獨編譯。

Reason

如果比JavaScript 更快、更簡單且類型安全會怎樣?

這就是創(chuàng)建Reason的 Facebook 開發(fā)者想要回答的問題。不過,他并沒有從頭開始構(gòu)建一種新語言,而是采用了 OCaml,并將其調(diào)整為類似于 JavaScript。

Reason使用項目 BucketScript編譯為 JavaScript,并且可以訪問 80% 的 JavaScript 工具和生態(tài)系統(tǒng)。它還可以編譯為準(zhǔn)系統(tǒng)、iOS、Android 和微控制器。

Red

Red是一種最初旨在克服 Rebol 語言限制的編程語言。Red 于 2011 年推出,受 Rebol、Lua 和 Scala 等語言的影響,對高級和低級編程都很有用。

該語言可用于開發(fā)從高級 GUI 到低級操作系統(tǒng)的所有方面。Red 擁有人性化的語法、低內(nèi)存占用和垃圾收集等優(yōu)點。

Rust

Rust解決了一些與 Go 相同的問題,如系統(tǒng)級別的線程和進(jìn)程安全,,但Rust 更像 C 風(fēng)格的語法

但Rust語言的缺點:靜態(tài)類型和缺乏垃圾收集

Rust可直接訪問內(nèi)存意味著程序員可以編寫低級代碼,如操作系統(tǒng)內(nèi)核。Rust 也非常適合嵌入式設(shè)備、網(wǎng)絡(luò)服務(wù)和命令行編寫。

對比 Go 語言,Rust 有什么優(yōu)勢和劣勢?

我并沒有什么編程的經(jīng)驗,覺得編程實在是太復(fù)雜了,不喜歡去研究太多,對這個也不怎么懂,只能說自己是個半吊子,就是所掌握的知識,也是東拼西湊的,朋友和我說點兒,自己去書上看一點兒,只能說根據(jù)自己的體驗給出一些體會吧。

Rust的優(yōu)勢是:

1、Rust把安全、精確的內(nèi)存管理作為一切的中心放在首要的位置。

2、Rust同時擁有特別強(qiáng)的控制性和特別強(qiáng)的安全性。

3、Rust語言通過: 優(yōu)秀的類型系統(tǒng)設(shè)計、 嚴(yán)格的編譯器靜態(tài)審查、 配合程序員局部核對、加上少量的運行時校驗,保障了內(nèi)存安全。

4、Rust的語言特別的復(fù)雜,導(dǎo)致學(xué)習(xí)曲線比較陡峭,對于初學(xué)者來說難度較大。但學(xué)通之后將終生受益。

5、效率高,速度特別的快

6、 支持范型

7、 社區(qū)活躍度很高,更加的強(qiáng)調(diào)了社區(qū)的作用。

8、Rust 有更強(qiáng)的語義,更容易捕獲錯誤的邏輯,編譯器直接檢查出你代碼中的不安全的部分

Rust的劣勢是:

1、 語言相對來說比較復(fù)雜,對于新手來說,讓新手摸不著頭腦。

2、還不算太穩(wěn)定。

其實我覺得什么代碼啊編程啊這些東西還是比較適合理工的學(xué)生去研究,我一看腦袋就大,完全不明白在講什么。我大概了解的就是這些,語言的話大家可以多方面的去了解,也不是說有缺點就是不好,看配置看個人吧,每個人習(xí)慣不一樣,也許有的人用不穩(wěn)定的還覺得挺好呢,有的人就喜歡比較完美的,在我看來編程這個東西真的是很復(fù)雜,會有很多的代碼,這些代碼弄得我自己頭都大了,有的時候還得去惡補(bǔ)一下。

Python , Go , Rust

Go語言非???,但并沒有為我提供足夠的類型安全。如果不需要并發(fā),那么我寧可繼續(xù)使用Python及其生態(tài)環(huán)境,而不會去冒險采用Go重寫一切。如果需要并發(fā),那我也不會使用Go,因為它缺乏類型安全,這個炸彈肯定會爆炸。

Rust非常適合并發(fā)和安全,盡管futures-rs的“箱子”(Rust里函數(shù)庫叫做“箱子”)依然非常新。我認(rèn)為,以后Rust會成為許多后臺的默認(rèn)語言。

駁狗屎文 "我為什么放棄Go語言

此篇文章流傳甚廣, 其實里面沒啥干貨, 而且里面很多觀點是有問題的. 這個文章在 golang-china 很早就討論過了.

最近因為 Rust 1.0 和 1.1 的發(fā)布, 導(dǎo)致這個文章又出來毒害讀者.

所以寫了這篇反駁文章, 指出其中的問題.

有好幾次,當(dāng)我想起來的時候,總是會問自己:我為什么要放棄Go語言?這個決定是正確的嗎?是明智和理性的嗎?其實我一直在認(rèn)真思考這個問題。

開門見山地說,我當(dāng)初放棄Go語言(golang),就是因為兩個“不爽”:第一,對Go語言本身不爽;第二,對Go語言社區(qū)里的某些人不爽。毫無疑問,這是非常主觀的結(jié)論。但是我有足夠詳實的客觀的論據(jù),用以支撐這個看似主觀的結(jié)論。

文末附有本文更新日志。

確實是非常主觀的結(jié)論, 因為里面有不少有問題的觀點(用來忽悠Go小白還行).

第0節(jié):我的Go語言經(jīng)歷

先說說我的經(jīng)歷吧,以避免被無緣無故地當(dāng)作Go語言的低級黑。

2009年底,Go語言(golang)第一個公開版本發(fā)布,籠罩著“Google公司制造”的光環(huán),吸引了許多慕名而來的嘗鮮者,我(Liigo)也身居其中,籠統(tǒng)的看了一些Go語言的資料,學(xué)習(xí)了基礎(chǔ)的教程,因?qū)ζ湔Z法中的分號和花括號不滿,很快就遺忘掉了,沒拿它當(dāng)一回事。

在2009年Go剛發(fā)布時, 確實是因為“Google公司制造”的光環(huán)而吸引了(包括文章作者和諸多IT記者)很多低級的嘗鮮者.

還好, 經(jīng)過5年的發(fā)展, 這些純粹因為光環(huán)來的投機(jī)者所剩已經(jīng)不多了(Google趨勢).

目前, 真正的Go用戶早就將Go用于實際的生產(chǎn)了.

說到 其語法中的分號和花括號不滿, 我想說這只是你的 個人主觀感受, 還有很多人對Go的分號和花括號很滿意,

包括水果公司的的 Swift 的語言設(shè)計者也很滿意這種風(fēng)格(Swift中的分號和花括號和Go基本相同).

如果只談 個人主觀感受, 我也可以說 Rust 的 fn 縮寫也很蛋疼!

兩年之后,2011年底,Go語言發(fā)布1.0的計劃被提上日程,相關(guān)的報道又多起來,我再次關(guān)注它,重新評估之后決定深入?yún)⑴cGo語言。我訂閱了其users、nuts、dev、commits等官方郵件組,堅持每天閱讀其中的電子郵件,以及開發(fā)者提交的每一次源代碼更新,給Go提交了許多改進(jìn)意見,甚至包括修改Go語言編譯器源代碼直接參與開發(fā)任務(wù)。如此持續(xù)了數(shù)月時間。

這個到是事實, 在 golang-china 有不少吵架的帖子, 感興趣的可以去挖下, 我就不展開說了.

到2012年初,Go 1.0發(fā)布,語言和標(biāo)準(zhǔn)庫都已經(jīng)基本定型,不可能再有大幅改進(jìn),我對Go語言未能在1.0定型之前更上一個臺階、實現(xiàn)自我突破,甚至帶著諸多明顯缺陷走向1.0,感到非常失望,因而逐漸疏遠(yuǎn)了它(所以Go 1.0之后的事情我很少關(guān)心)。后來看到即將發(fā)布的Go 1.1的Release Note,發(fā)現(xiàn)語言層面沒有太大改變,只是在庫和工具層面有所修補(bǔ)和改進(jìn),感到它尚在幼年就失去成長的動力,越發(fā)失望。外加Go語言社區(qū)里的某些人,其中也包括Google公司負(fù)責(zé)開發(fā)Go語言的某些人,其態(tài)度、言行,讓我極度厭惡,促使我決絕地離棄Go語言。

真的不清楚樓主說的可以在 Go1.0 之前短時間內(nèi)能實現(xiàn)的 重大改進(jìn)和諸多明顯缺陷 是什么.

如果是樓主說前面的 其語法中的分號和花括號不滿 之類的重大改進(jìn), 我只能說這只是你的 個人主觀感受 而已,

你的很多想法只能說服你自己, 沒辦法說服其他絕大部分人(不要以為像C++或Rust那樣什么特性都有就NB了, 各種NB特性加到一起只能是 要你命3000, 而絕對不會是什么 銀彈).

Go 1.1的Release Note,發(fā)現(xiàn)語言層面沒有太大改變. 語言層沒有改變是是因為 Go1 作出的向后兼容的承諾. 對于工業(yè)級的語言來說, Go1 這個只能是優(yōu)點. 如果連語言層在每個版本都會出現(xiàn)諸多大幅改進(jìn), 那誰還敢用Go語言來做生產(chǎn)開發(fā)呢(我承認(rèn)Rust的改動很大膽, 但也說明了Rust還處于比較幼稚和任性的階段)?

說 Go語言社區(qū)里的某些人固執(zhí) 的觀點我是同意的. 但是這些 固執(zhí) 的人是可以講道理的, 但是他們對很多東西的要求很高(特別是關(guān)于Go的設(shè)計哲學(xué)部分).

只要你給的建議有依據(jù)(語言的設(shè)計哲學(xué)是另外一回事情), 他們絕對不會盲目的拒絕(只是討論的周期會比較長).

關(guān)于樓主提交的給Go文件添加BOM的文章, 需要補(bǔ)充說明下.

在Go1.0發(fā)布的時候, Go語言的源文件(.go)明確要求必須是UTF8編碼的, 而且是無BOM的UTF8編碼的.

注意: 這個 無BOM的UTF8編碼 的限制僅僅是 針對 Go語言的源文件(.go).

這個限制并不是說不允許用戶處理帶BOM的UTF8的txt文件!

我覺得對于寫Go程序來說, 這個限制是沒有任何問題的, 到目前為止, 我還從來沒有使用過帶BOM的.go文件.

不僅是因為帶BOM的.go文件沒有太多的意義, 而且有很多的缺陷.

BOM的原意是用來表示編碼是大端還是小端的, 主要用于UTF16和UTF32. 對于 UTF8 來說, BOM 沒有任何存在的意義(正是Go的2個作者發(fā)明了UTF8, 徹底解決了全球的編碼問題).

但是, 在現(xiàn)實中, 因為MS的txt記事本, 對于中文環(huán)境會將txt(甚至是C/C++源文件)當(dāng)作GBK編碼(GBK是個爛編碼),

為了區(qū)別到底是GBK還是UTF8, MS的記事本在前面加了BOM這個垃圾(被GBK占了茅坑), 這里的bom已經(jīng)不是表示字節(jié)序本意了. 不知道有沒有人用ms的記事本寫網(wǎng)頁, 然后生成一個帶bom的utf8網(wǎng)頁肯定很有意思.

這是MS的記事本的BUG: 它不支持生成無BOM的UTF8編碼的文本文件!

這些是現(xiàn)實存在的帶BOM的UTF8編碼的文本文件, 但是它們肯定都不是Go語言源文件!

所以說, Go語言的源文件即使強(qiáng)制限制了無BOM的UTF8編碼要求, 也是沒有任何問題的(而且我還希望有這個限制).

雖然后來Go源文件接受帶BOM的UTF8了, 但是運行 go fmt 之后, 還是會刪除掉BOM的(因為BOM就是然并卵). 也就是說 帶 BOM 的 Go 源文件是不符合 Go語言的編碼風(fēng)格的, go fmt 會強(qiáng)制刪除 BOM 頭.

前面說了BOM是MS帶來的垃圾, 但是BOM的UTF8除了然并卵之外還有很多問題, 因為BOM在string的開頭嵌入了垃圾,

導(dǎo)致正則表達(dá)式, string的鏈接運算等操作都被會被BOM這個垃圾所污染. 對于.go語言, 即使代碼完全一樣, 有BOM和無BOM會導(dǎo)致文件的MD5之類的校驗碼不同.

所以, 我覺得Go用戶不用糾結(jié)BOM這個無關(guān)緊要的東西.

在上一個10年,我(Liigo)在我所屬的公司里,深度參與了兩個編程語言項目的開發(fā)。我想,對于如何判斷某個編程語言的優(yōu)劣,或者說至少對于如何判斷某個編程語言是否適合于我自己,我應(yīng)該還是有一點發(fā)言權(quán)的。

第1節(jié):我為什么對Go語言不爽?

Go語言有很多讓我不爽之處,這里列出我現(xiàn)在還能記起的其中一部分,排名基本上不分先后。讀者們耐心地看完之后,還能淡定地說一句“我不在乎”嗎?

1.1 不允許左花括號另起一行

關(guān)于對花括號的擺放,在C語言、C++、Java、C#等社區(qū)中,十余年來存在持續(xù)爭議,從未形成一致意見。在我看來,這本來就是主觀傾向很重的抉擇,不違反原則不涉及是非的情況下,不應(yīng)該搞一刀切,讓程序員或團(tuán)隊自己選擇就足夠了。編程語言本身強(qiáng)行限制,把自己的喜好強(qiáng)加給別人,得不償失。無論傾向于其中任意一種,必然得罪與其對立的一群人。雖然我現(xiàn)在已經(jīng)習(xí)慣了把左花括號放在行尾,但一想到被禁止其他選擇,就感到十分不爽。Go語言這這個問題上,沒有做到“團(tuán)結(jié)一切可以團(tuán)結(jié)的力量”不說,還有意給自己樹敵,太失敗了。

我覺得Go最偉大的發(fā)明是 go fmt, 從此Go用戶不會再有花括弧的位置這種無聊爭論了(當(dāng)然也少了不少灌水和上tiobe排名的機(jī)會).

是這優(yōu)點, Swift 語言也使用和 Go 類似的風(fēng)格(當(dāng)然樓主也可能鄙視swift的作者).

1.2 編譯器莫名其妙地給行尾加上分號

對Go語言本身而言,行尾的分號是可以省略的。但是在其編譯器(gc)的實現(xiàn)中,為了方便編譯器開發(fā)者,卻在詞法分析階段強(qiáng)行添加了行尾的分號,反過來又影響到語言規(guī)范,對“怎樣添加分號”做出特殊規(guī)定。這種變態(tài)做法前無古人。在左花括號被意外放到下一行行首的情況下,它自動在上一行行尾添加的分號,會導(dǎo)致莫名其妙的編譯錯誤(Go 1.0之前),連它自己都解釋不明白。如果實在處理不好分號,干脆不要省略分號得了;或者,Scala和JavaScript的編譯器是開源的,跟它們學(xué)學(xué)怎么處理省略行尾分號可以嗎?

又是樓主的 個人主觀感受, 不過我很喜歡這個特性. Swift 語言也是類似.

1.3 極度強(qiáng)調(diào)編譯速度,不惜放棄本應(yīng)提供的功能

程序員是人不是神,編碼過程中免不了因為大意或疏忽犯一些錯。其中有一些,是大家集體性的很容易就中招的錯誤(Go語言里的例子我暫時想不起來,C++里的例子有“基類析構(gòu)函數(shù)不是虛函數(shù)”)。這時候編譯器應(yīng)該站出來,多做一些檢查、約束、核對性工作,盡量阻止常規(guī)錯誤的發(fā)生,盡量不讓有潛在錯誤的代碼編譯通過,必要時給出一些警告或提示,讓程序員留意。編譯器不就是機(jī)器么,不就是應(yīng)該多做臟活累活雜活、減少人的心智負(fù)擔(dān)么?編譯器多做一項檢查,可能會避免數(shù)十萬程序員今后多年內(nèi)無數(shù)次犯同樣的錯誤,節(jié)省的時間不計其數(shù),這是功德無量的好事。但是Go編譯器的作者們可不這么想,他們不愿意自己多花幾個小時給編譯器增加新功能,覺得那是虧本,反而減慢了編譯速度。他們以影響編譯速度為由,拒絕了很多對編譯器改進(jìn)的要求。典型的因噎廢食。強(qiáng)調(diào)編譯速度固然值得贊賞,但如果因此放棄應(yīng)有的功能,我不贊成。

編譯速度是很重要的, 如果編譯速度夠慢, 語言再好也不會有人使用的.

比如C/C++的增量編譯/預(yù)編譯頭文件/并發(fā)編譯都是為了提高編譯速度.

Rust1.1 也號稱 比 1.0 的編譯時間減少了32% (注意: 不是運行速度).

當(dāng)然, Go剛面世的時候, 編譯速度是其中的一個設(shè)計目標(biāo).

不過我想樓主, 可能想說的是因為編譯器自己添加分號而導(dǎo)致的編譯錯誤的問題.

我覺得Go中 { 不能另起一行是語言特性, 如果修復(fù)這個就是引入了新的錯誤.

其他的我真想不起來還有哪些 調(diào)編譯速度,不惜放棄本應(yīng)提供的功能 (不要提泛型, 那是因為還沒有好的設(shè)計).

1.4 錯誤處理機(jī)制太原始

在Go語言中處理錯誤的基本模式是:函數(shù)通常返回多個值,其中最后一個值是error類型,用于表示錯誤類型極其描述;調(diào)用者每次調(diào)用完一個函數(shù),都需要檢查這個error并進(jìn)行相應(yīng)的錯誤處理:if err != nil { /*這種代碼寫多了不想吐么*/ }。此模式跟C語言那種很原始的錯誤處理相比如出一轍,并無實質(zhì)性改進(jìn)。實際應(yīng)用中很容易形成多層嵌套的if else語句,可以想一想這個編碼場景:先判斷文件是否存在,如果存在則打開文件,如果打開成功則讀取文件,如果讀取成功再寫入一段數(shù)據(jù),最后關(guān)閉文件,別忘了還要處理每一步驟中出現(xiàn)錯誤的情況,這代碼寫出來得有多變態(tài)、多丑陋?實踐中普遍的做法是,判斷操作出錯后提前return,以避免多層花括號嵌套,但這么做的后果是,許多錯誤處理代碼被放在前面突出的位置,常規(guī)的處理邏輯反而被掩埋到后面去了,代碼可讀性極差。而且,error對象的標(biāo)準(zhǔn)接口只能返回一個錯誤文本,有時候調(diào)用者為了區(qū)分不同的錯誤類型,甚至需要解析該文本。除此之外,你只能手工強(qiáng)制轉(zhuǎn)換error類型到特定子類型(靜態(tài)類型的優(yōu)勢沒了)。至于panic - recover機(jī)制,致命的缺陷是不能跨越庫的邊界使用,注定是一個半成品,最多只能在自己的pkg里面玩一玩。Java的異常處理雖然也有自身的問題(比如Checked Exceptions),但總體上還是比Go的錯誤處理高明很多。

話說, 軟件開發(fā)都發(fā)展了半個世紀(jì), 還是無實質(zhì)性改進(jìn). 不要以為弄一個異常的語法糖就是革命了.

我只能說錯誤和異常是2個不同的東西, 將所有錯誤當(dāng)作異常那是SB行為.

正因為有異常這個所謂的銀彈, 導(dǎo)致很多等著別人幫忙擦屁股的行為(注意 shit 函數(shù)拋出的絕對不會是一種類型的 shit, 而被其間接調(diào)用的各種 xxx_shit 也可能拋出各種類型的異常, 這就導(dǎo)致 catch 失控了):

int main() {

try {

shit();

} catch( /* 到底有幾千種 shit ? */) {

...

}

}

Go的建議是 panic - recover 不跨越邊界, 也就是要求正常的錯誤要由pkg的處理掉.

這是負(fù)責(zé)任的行為.

再說Go是面向并發(fā)的編程語言, 在海量的 goroutine 中使用 try/catch 是不是有一種不倫不類的感覺呢?

1.5 垃圾回收器(GC)不完善、有重大缺陷

在Go 1.0前夕,其垃圾回收器在32位環(huán)境下有內(nèi)存泄漏,一直拖著不肯改進(jìn),這且不說。Go語言垃圾回收器真正致命的缺陷是,會導(dǎo)致整個進(jìn)程不可預(yù)知的間歇性停頓。像某些大型后臺服務(wù)程序,如游戲服務(wù)器、APP容器等,由于占用內(nèi)存巨大,其內(nèi)存對象數(shù)量極多,GC完成一次回收周期,可能需要數(shù)秒甚至更長時間,這段時間內(nèi),整個服務(wù)進(jìn)程是阻塞的、停頓的,在外界看來就是服務(wù)中斷、無響應(yīng),再牛逼的并發(fā)機(jī)制到了這里統(tǒng)統(tǒng)失效。垃圾回收器定期啟動,每次啟動就導(dǎo)致短暫的服務(wù)中斷,這樣下去,還有人敢用嗎?這可是后臺服務(wù)器進(jìn)程,是Go語言的重點應(yīng)用領(lǐng)域。以上現(xiàn)象可不是我假設(shè)出來的,而是事實存在的現(xiàn)實問題,受其嚴(yán)重困擾的也不是一家兩家了(2013年底ECUG Con 2013,京東的劉奇提到了Go語言的GC、defer、標(biāo)準(zhǔn)庫實現(xiàn)是性能殺手,最大的痛苦是GC;美團(tuán)的沈鋒也提到Go語言的GC導(dǎo)致后臺服務(wù)間隔性停頓是最大的問題。更早的網(wǎng)絡(luò)游戲仙俠道開發(fā)團(tuán)隊也曾受Go垃圾回收的沉重打擊)。在實踐中,你必須努力減少進(jìn)程中的對象數(shù)量,以便把GC導(dǎo)致的間歇性停頓控制在可接受范圍內(nèi)。除此之外你別無選擇(難道你還想自己更換GC算法、甚至砍掉GC?那還是Go語言嗎?)。跳出圈外,我近期一直在思考,一定需要垃圾回收器嗎?沒有垃圾回收器就一定是歷史的倒退嗎?(可能會新寫一篇博客文章專題探討。)

這是說的是32位系統(tǒng), 這絕對不是Go語言的重點應(yīng)用領(lǐng)域!! 我可以說Go出生就是面向64位系統(tǒng)和多核心CPU環(huán)境設(shè)計的. (再說 Rust 目前好像還不支持 XP 吧, 這可不可以算是影響巨大?)

32位當(dāng)時是有問題, 但是對實際生產(chǎn)影響并不大(請問樓主還是在用32位系統(tǒng)嗎, 還只安裝4GB的內(nèi)存嗎). 如果是8位單片機(jī)環(huán)境, 建議就不要用Go語言了, 直接C語言好了.

而且這個問題早就不存在了(大家可以去看Go的發(fā)布日志).

Go的出生也就5年時間, GC的完善和改進(jìn)是一個持續(xù)的工作, 2015年8月將發(fā)布的 Go1.5將采用并行GC.

關(guān)于GC的被人詬病的地方是會導(dǎo)致卡頓, 但是我以為這個主要是因為GC的實現(xiàn)還不夠完美而導(dǎo)致的.

如果是完美的并發(fā)和增量的GC, 那應(yīng)該不會出現(xiàn)大的卡頓問題的.

當(dāng)然, 如果非要實時性, 那用C好了(實時并不表示性能高, 只是響應(yīng)時間可控).

對于Rust之類沒有GC的語言來說, 想很方便的開發(fā)并發(fā)的后臺程序那幾乎是不可能的.

不要總是吹Rust能代替底層/中層/上層的開發(fā), 我們要看有誰用Rust真的做了什么.

1.6 禁止未使用變量和多余import

Go編譯器不允許存在被未被使用的變量和多余的import,如果存在,必然導(dǎo)致編譯錯誤。但是現(xiàn)實情況是,在代碼編寫、重構(gòu)、調(diào)試過程中,例如,臨時性的注釋掉一行代碼,很容易就會導(dǎo)致同時出現(xiàn)未使用的變量和多余的import,直接編譯錯誤了,你必須相應(yīng)的把變量定義注釋掉,再翻頁回到文件首部把多余的import也注釋掉,……等事情辦完了,想把剛才注釋的代碼找回來,又要好幾個麻煩的步驟。還有一個讓人蛋疼的問題,編寫數(shù)據(jù)庫相關(guān)的代碼時,如果你import某數(shù)據(jù)庫驅(qū)動的pkg,它編譯給你報錯,說不需要import這個未被使用的pkg;但如果你聽信編譯器的話刪掉該import,編譯是通過了,運行時必然報錯,說找不到數(shù)據(jù)庫驅(qū)動;你看看程序員被折騰的兩邊不是人,最后不得不請出大神:import _。對待這種問題,一個比較好的解決方案是,視其為編譯警告而非編譯錯誤。但是Go語言開發(fā)者很固執(zhí),不容許這種折中方案。

這個問題我只能說樓主的吐槽真的是沒水平.

為何不使用的是錯誤而不是警告? 這是為了將低級的bug消滅在編譯階段(大家可以想下C/C++的那么多警告有什么卵用).

而且, import 即使沒有使用的話, 也是用副作用的, 因為 import 會導(dǎo)致 init 和全局變量的初始化.

如果某些代碼沒有使用, 為何要執(zhí)行 init 這些初始化呢?

如果是因為調(diào)試而添加的變量, 那么調(diào)試完刪除不是很正常的要求嗎?

如果是因為調(diào)試而要導(dǎo)入fmt或log之類的包, 刪除調(diào)試代碼后又導(dǎo)致 import 錯誤的花,

樓主難道不知道在一個獨立的文件包裝下類似的輔助調(diào)試的函數(shù)嗎?

import (

"fmt"

"log"

)

func logf(format string, a ...interface{}) {

file, line := callerFileLine()

fmt.Fprintf(os.Stderr, "%s:%d: ", file, line)

fmt.Fprintf(os.Stderr, format, a...)

}

func fatalf(format string, a ...interface{}) {

file, line := callerFileLine()

fmt.Fprintf(os.Stderr, "%s:%d: ", file, line)

fmt.Fprintf(os.Stderr, format, a...)

os.Exit(1)

}

import _ 是有明確行為的用法, 就是為了執(zhí)行包中的 init 等函數(shù)(可以做某些注冊操作).

將警告當(dāng)作錯誤是Go的一個哲學(xué), 當(dāng)然在樓主看來這是白癡做法.

1.7 創(chuàng)建對象的方式太多令人糾結(jié)

創(chuàng)建對象的方式,調(diào)用new函數(shù)、調(diào)用make函數(shù)、調(diào)用New方法、使用花括號語法直接初始化結(jié)構(gòu)體,你選哪一種?不好選擇,因為沒有一個固定的模式。從實踐中看,如果要創(chuàng)建一個語言內(nèi)置類型(如channel、map)的對象,通常用make函數(shù)創(chuàng)建;如果要創(chuàng)建標(biāo)準(zhǔn)庫或第三方庫定義的類型的對象,首先要去文檔里找一下有沒有New方法,如果有就最好調(diào)用New方法創(chuàng)建對象,如果沒有New方法,則退而求其次,用初始化結(jié)構(gòu)體的方式創(chuàng)建其對象。這個過程頗為周折,不像C++、Java、C#那樣直接new就行了。

C++的new是狗屎. new導(dǎo)致的問題是構(gòu)造函數(shù)和普通函數(shù)的行為不一致, 這個補(bǔ)丁特性真的沒啥優(yōu)越的.

我還是喜歡C語言的 fopen 和 malloc 之類構(gòu)造函數(shù), 構(gòu)造函數(shù)就是普通函數(shù), Go語言中也是這樣.

C++中, 除了構(gòu)造不兼容普通函數(shù), 析構(gòu)函數(shù)也是不兼容普通函數(shù). 這個而引入的坑有很多吧.

1.8 對象沒有構(gòu)造函數(shù)和析構(gòu)函數(shù)

沒有構(gòu)造函數(shù)還好說,畢竟還有自定義的New方法,大致也算是構(gòu)造函數(shù)了。沒有析構(gòu)函數(shù)就比較難受了,沒法實現(xiàn)RAII。額外的人工處理資源清理工作,無疑加重了程序員的心智負(fù)擔(dān)。沒人性啊,還嫌我們程序員加班還少嗎?C++里有析構(gòu)函數(shù),Java里雖然沒有析構(gòu)函數(shù)但是有人家finally語句啊,Go呢,什么都沒有。沒錯,你有個defer,可是那個defer問題更大,詳見下文吧。

defer 可以覆蓋析構(gòu)函數(shù)的行為, 當(dāng)然 defer 還有其他的任務(wù). Swift2.0 也引入了一個簡化版的 defer 特性.

1.9 defer語句的語義設(shè)定不甚合理

Go語言設(shè)計defer語句的出發(fā)點是好的,把釋放資源的“代碼”放在靠近創(chuàng)建資源的地方,但把釋放資源的“動作”推遲(defer)到函數(shù)返回前執(zhí)行。遺憾的是其執(zhí)行時機(jī)的設(shè)置似乎有些不甚合理。設(shè)想有一個需要長期運行的函數(shù),其中有無限循環(huán)語句,在循環(huán)體內(nèi)不斷的創(chuàng)建資源(或分配內(nèi)存),并用defer語句確保釋放。由于函數(shù)一直運行沒有返回,所有defer語句都得不到執(zhí)行,循環(huán)過程中創(chuàng)建的大量短暫性資源一直積累著,得不到回收。而且,系統(tǒng)為了存儲defer列表還要額外占用資源,也是持續(xù)增加的。這樣下去,過不了多久,整個系統(tǒng)就要因為資源耗盡而崩潰。像這類長期運行的函數(shù),http.ListenAndServe()就是典型的例子。在Go語言重點應(yīng)用領(lǐng)域,可以說幾乎每一個后臺服務(wù)程序都必然有這么一類函數(shù),往往還都是程序的核心部分。如果程序員不小心在這些函數(shù)中使用了defer語句,可以說后患無窮。如果語言設(shè)計者把defer的語義設(shè)定為在所屬代碼塊結(jié)束時(而非函數(shù)返回時)執(zhí)行,是不是更好一點呢?可是Go 1.0早已發(fā)布定型,為了保持向后兼容性,已經(jīng)不可能改變了。小心使用defer語句!一不小心就中招。

前面說到 defer 還有其他的任務(wù), 也就是 defer 中執(zhí)行的 recover 可以捕獲 panic 拋出的異常.

還有 defer 可以在 return 之后修改命名的返回值.

上面2個工作要求 defer 只能在函數(shù)退出時來執(zhí)行.

樓主說的 defer 是類似 Swift2.0 中 defer 的行為, 但是 Swift2.0 中 defer 是沒有前面2個特性的.

Go中的defer是以函數(shù)作用域作為觸發(fā)的條件的, 是會導(dǎo)致樓主說的在 for 中執(zhí)行的錯誤用法(哪個語言沒有坑呢?).

不過 for 中 局部 defer 也是有辦法的 (Go中的defer是以函數(shù)作用域):

for {

func(){

f, err := os.Open(...)

defer f.Close()

}()

}

在 for 中做一個閉包函數(shù)就可以了. 自己不會用不要怪別人沒告訴你.

1.10 許多語言內(nèi)置設(shè)施不支持用戶定義的類型

for in、make、range、channel、map等都僅支持語言內(nèi)置類型,不支持用戶定義的類型(?)。用戶定義的類型沒法支持for in循環(huán),用戶不能編寫像make、range那樣“參數(shù)類型和個數(shù)”甚至“返回值類型和個數(shù)”都可變的函數(shù),不能編寫像channel、map那樣類似泛型的數(shù)據(jù)類型。語言內(nèi)置的那些東西,處處充斥著斧鑿的痕跡。這體現(xiàn)了語言設(shè)計的局限性、封閉性、不完善,可擴(kuò)展性差,像是新手作品——且不論其設(shè)計者和實現(xiàn)者如何權(quán)威。延伸閱讀:Go語言是30年前的陳舊設(shè)計思想,用戶定義的東西幾乎都是二等公民(Tikhon Jelvis)。

說到底, 這個是因為對泛型支持的不完備導(dǎo)致的.

Go語言是沒啥NB的特性, 但是Go的特性和工具組合在一起就是好用.

這就是Go語言NB的地方.

1.11 沒有泛型支持,常見數(shù)據(jù)類型接口丑陋

沒有泛型的話,List、Set、Tree這些常見的基礎(chǔ)性數(shù)據(jù)類型的接口就只能很丑陋:放進(jìn)去的對象是一個具體的類型,取出來之后成了無類型的interface{}(可以視為所有類型的基礎(chǔ)類型),還得強(qiáng)制類型轉(zhuǎn)換之后才能繼續(xù)使用,令人無語。Go語言缺少min、max這類函數(shù),求數(shù)值絕對值的函數(shù)abs只接收/返回雙精度小數(shù)類型,排序接口只能借助sort.Interface無奈的回避了被比較對象的類型,等等等等,都是沒有泛型導(dǎo)致的結(jié)果。沒有泛型,接口很難優(yōu)雅起來。Go開發(fā)者沒有明確拒絕泛型,只是說還沒有找到很好的方法實現(xiàn)泛型(能不能學(xué)學(xué)已經(jīng)開源的語言呀)?,F(xiàn)實是,Go 1.0已經(jīng)定型,泛型還沒有,那些丑陋的接口為了保持向后兼容必須長期存在著。

Go有自己的哲學(xué), 如果能有和目前哲學(xué)不沖突的泛型實現(xiàn), 他們是不會反對的.

如果只是簡單學(xué)學(xué)(或者叫抄襲)已經(jīng)開源的語言的語法, 那是C++的設(shè)計風(fēng)格(或者說C++從來都是這樣設(shè)計的, 有什么特性就抄什么), 導(dǎo)致了各種腦裂的編程風(fēng)格.

編譯時泛型和運行時泛型可能是無法完全兼容的, 看這個例子:

type AdderT interface {

Add(a, b T) T

}

網(wǎng)站名稱:go語言與rust,go語言與java的區(qū)別
標(biāo)題來源:http://muchs.cn/article10/hcipdo.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站導(dǎo)航、網(wǎng)站策劃關(guān)鍵詞優(yōu)化、營銷型網(wǎng)站建設(shè)動態(tài)網(wǎng)站、做網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

網(wǎng)站優(yōu)化排名