前言
學習一門新的語言肯定是要從他的基本語法開始,語法構成了整個程序設計的基礎,從語法中我們也可以看到這門語言的一些特性,但是話說回來,語法這東西,不同的語言大同小異,所以這也對語法的記憶造成了一定的難度,其實最好的方法應該是旁邊有本書,隨時可以拿過來查閱或者糾正。
Go 的初學者可能會有這樣的疑問:為什么 Go 的聲明語法與傳統的其他 C 家族編程語言不太一樣?在這篇文章中我們會比較這兩種不同的方式,并且也會解釋為什么。下面話不多說了,來一起看看詳細的介紹吧。
C 變量
首先,讓我們說說 C 中的語法。C 使用了一種不尋常的巧妙的方法來實現聲明語法。我們不是用什么特殊的語法來描述類型,而是寫一個表達式,這個表達式包含兩個部分:被聲明的變量和變量的類型。
上面這行代碼聲明了一個類型為 int 的變量 x。一般來說,為了弄清楚如何編寫新變量的類型,可以先寫一個含基本類型變量的表達式,然后將基本類型放在左邊,將表達式放在右邊。
因此,下面的聲明:
描述的是 p 是一個指向 int 類型的指針,因為 ‘*p' 的類型為 int。而 a 是一個 int 類型的數組,因為 ‘a[3]' (這里請忽略下標的值 3,它只是說明數組的大小)的類型是 int。
那函數呢?在最開始的時候,C 的函數聲明是將 參數的類型寫在括號外面的,像這樣:
int main(argc, argv)
int argc;
char *argv[];
{ /* ... */ }
再一次,我們可以看到 main 是一個函數,因為表達式 main(argc, argv) 返回了一個 int 類型的值。現在大家比較習慣寫成這樣:
int main(int argc, char *argv[]) { /* ... */ }
但是基本的結構還是一樣的。
對于簡單的類型來說這種巧妙的語法思想是能很好工作的,但是一旦類型變得復雜就會令人感到困惑了。非常經典的一個例子就是聲明一個函數指針。遵循著規則,你得到了下面的這種寫法:
fp 是一個指向函數的指針,因為如果你寫一個表達式 (*fp)(a, b) 你會調用函數并得到一個 int 類型的值。那如果 fp 的其中一個入參它本身也是一個函數呢?
int (*fp)(int (*ff)(int x, int y), int b)
這就變得開始難以閱讀了。
當然,我們可以在聲明一個函數的時候去掉參數名,那么 main 函數可以聲明成:
讓我們回想一下,argv 是這樣聲明的,
通過把變量名放在中間來聲明類似 char *[] 這樣類型的時候其實是令人困惑的。
然后我們再來看看如果我們將入參變量名去掉的情況下 fp 函數的聲明是怎么樣的:
int (*fp)(int (*)(int, int), int)
無論將變量名放在內部的哪里都不那么清晰明了。對于第一個入參:
我想這不太容易能一眼看出是在聲明一個指向函數的指針。再進一步,如果我們的返回值也是一個函數指針呢?
int (*(*fp)(int (*)(int, int), int))(int, int)
這根本就看不清聲明出來的 fp 到底是個啥玩意。。。
你自己也可以構造出更多這類詳細的例子,但是這些都說明了 C 的聲明語法可能引入的一些困難。
不過還有一點需要提出。因為類型和聲明的語法是相同的,所以解析中間類型的表達式是很困難的。這就是為什么 C 的類型轉換總是用括號括起來:
Go 語法
非 C 家族的編程語言通常使用不同的聲明類型的語法:變量名通常放在前面,然后緊跟著一個冒號。因此我們上面的例子就變成了這樣:
x: int
p: pointer to int
a: array[3] of int
這些聲明是明確的,如果從左往右讀你會發現也是詳細的。Go 語言從中得到了啟發,但為了簡潔起見,刪除了冒號和一些關鍵字:
這個例子中 [3]int 與如何在表達式中使用 a 這兩者似乎沒有直接的對應。(后面一小節中我們會講到指針的。)你可以通過單獨的語法來獲得清晰的結果。
現在讓我們考慮下函數。讓我們把這個聲明寫成 Go 的形式,盡管在 Go 中真正的 main 函數是沒有入參的:
func main(argc int, argv []string) int
表面上這和 C 語言并沒什么不同,除了將字符數組改成了字符串形式。但是從左往右讀起來卻很順暢:
函數 main 需要傳入一個整型和字符串切片并且返回一個整型。(譯者注:直到譯者看到這篇文章,譯者才發現原來這么寫讀起來竟這么順暢。。。)
即便舍去變量名還是很明確——因為對于類型聲明上沒有位置的變化,所以也沒有什么困惑。
func main(int, []string) int
這種從左到右的風格有一個優點:就算類型變得越來越復雜,這種方式還是表現得很得當。
舉個聲明函數變量的例子(類似在 C 語言中的函數指針):
f func(func(int, int) int, int) int
或者如果 f 返回的也是一個函數(譯者注:邊寫邊讀你會再次驚訝于這絲滑般的順暢感。。。):
f func(func(int, int) int, int) func(int, int) int
從左到右依然讀起來很順暢,并且當變量名被聲明的時候也很明顯。
類型和表達式的語法的不同點使得在 Go 中編寫和調用閉包是那么的簡單:
sum := func(a, b int) int { return a + b } (3, 4)
指針
指針這家伙總是表現得“與眾不同”一點。觀察下數組和切片,舉個例子,Go 的類型語法將方括號放在類型的左邊,但是賦值表達式語法卻是將其放在表達式的右邊:
為了讓大家有一種熟悉的感覺,Go 的指針同樣延續 C 語言中的 * 符號,但是我們不能簡單的將指針類型也反轉一下。所以指針使用方式如下:
我們不能簡單粗暴地改成這樣:
因為后綴 會與乘法的 相混淆。那或許我們可以使用 ^,舉個例子:
但同樣的這個符號也已經有其他含義了,類型和表達式在前綴后綴的問題上總是在許多方面使事情復雜化。舉個例子,
這是一種寫法,但一旦以 * 打頭就必須用括號將其包住:
如果我們愿意放棄 * 作為指針語法,那么這些括號就不是必要的了。(譯者注:但還能有更好的指針語法嗎。。。)
所以 Go 的指針語法與熟悉的 C 語言是類似的,但這個關聯也意味著我們不得不使用括號來消除語法中的類型和表達式之間的差異。
總體而言,我們相信 Go 的類型語法比 C 的要更容易理解,尤其是當事情變得復雜的時候。
關于Go語言為何要采用這種倒序語法呢?
Go的設計者Rob Pike的一篇介紹Go聲明語法的文章給出了答案,其中談到了Go聲明語法的設計考量。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
您可能感興趣的文章:- Go語言中的Slice學習總結
- GO 語言學習指南
- Go語言學習筆記之反射用法詳解
- Go語言基礎學習教程
- Go語言函數學習教程
- golang新手不注意可能會出現的一些小問題