除了Java、Python和JavaScript之外,如果要開始學(xué)習(xí)一門新語(yǔ)言的話,我想應(yīng)該是Go!
萊蕪ssl適用于網(wǎng)站、小程序/APP、API接口等需要進(jìn)行數(shù)據(jù)傳輸應(yīng)用場(chǎng)景,ssl證書未來(lái)市場(chǎng)廣闊!成為創(chuàng)新互聯(lián)的ssl證書銷售渠道,可以享受市場(chǎng)價(jià)格4-6折優(yōu)惠!如果有意向歡迎電話聯(lián)系或者加微信:13518219792(備注:SSL證書合作)期待與您的合作!
Go語(yǔ)言正在被越來(lái)越多的公司使用。我們公司的后端服務(wù)已經(jīng)全面采用Go語(yǔ)言實(shí)現(xiàn)了。
最開始接觸Go語(yǔ)言是去年將一份Go代碼“翻譯”成Python并集成到測(cè)試平臺(tái)上,說(shuō)來(lái)也挺神奇,我從來(lái)沒(méi)學(xué)過(guò)Go卻完成了這個(gè)工作,這也側(cè)面反應(yīng)了Go的語(yǔ)法還是很平易近人的。
今年,在海翔飛調(diào)崗之后已經(jīng)沒(méi)有太多時(shí)間寫代碼了,但如果要開始學(xué)習(xí)一個(gè)新的語(yǔ)言或技術(shù)的話,我最想學(xué)的是Go!
目前來(lái)看,Go似乎還并沒(méi)有太多測(cè)試人員使用的場(chǎng)景,不過(guò),我之前介紹過(guò)的BDD行為驅(qū)動(dòng)框架gauge是由Go開發(fā)的,當(dāng)然,它也支持使用Go來(lái)編寫B(tài)DD測(cè)試代碼。
對(duì)于,已經(jīng)有一定開發(fā)經(jīng)驗(yàn)的同學(xué),如何快速的開始學(xué)習(xí)Go語(yǔ)言呢?我這里給一些思路。
#### 第一步:下載和安裝
在配置環(huán)境的時(shí)候你需要重點(diǎn)了解GOROOT、GOPATH的作用。
你還要準(zhǔn)備一款稱手的編輯器,如果你像我一樣,一直都在使用VS Code的話,那么就它就可以了。
#### 第二步:從hello world開始
先運(yùn)行一個(gè)hello world程序,認(rèn)識(shí)Go語(yǔ)言的語(yǔ)法。
package main
import ( ? ?"fmt")
func main(){
fmt.Println("helloworld!")
}
#### 第三步:熟悉Go的語(yǔ)法
接下來(lái),你可能要花一周左右的時(shí)間熟悉Go語(yǔ)言的語(yǔ)法。比如,變量定義、if/for、函數(shù)、Map、跨文件的程序調(diào)用…等,當(dāng)然,還有一些Go特有的知識(shí)。
當(dāng)然,我更喜歡看視頻教程,雖然質(zhì)量參差不齊,但我仍然覺(jué)得看視頻比我自己看書更有效率。
熟悉一段Go代碼:
package main
import"fmt"
func myFunc() {
i := 0
Here: ? //這行的第一個(gè)詞,以冒號(hào)結(jié)束作為標(biāo)簽
fmt.Println(i)
i++ ? ?if i 10{ ? ? ? ?goto Here ? //跳轉(zhuǎn)到Here去
}
}func main() { ? ?//調(diào)用函數(shù)
myFunc()
}
#### 第四步:Go如何做單元測(cè)試
針對(duì)Go做測(cè)試也非常簡(jiǎn)單。比如,這是一個(gè)被測(cè)試文件:add.go。
package test_demofunc Add(a int, b int) int{ ? ?return a + b
}
下面針對(duì)Add()函數(shù)編寫測(cè)試用例,test_add.go
package test_demo
import ( ? ?"testing")
func TestAdd1(t *testing.T){
r:= Add(1, 2) ? ?if r !=3{
t.Errorf("Add(1, 2)failed. Got %d, expected 3.", r)
}
}
func TestAdd2(t *testing.T){
r:= Add(2, 2) ? ?if r !=4{
t.Errorf("Add(2, 2)failed. Got %d, expected 4.", r)
}
}
你只需要執(zhí)行?go test?命令就可以運(yùn)行上面的測(cè)試了。
#### 第五步:從哪兒找第三方庫(kù)
當(dāng)然,你只學(xué)習(xí)go語(yǔ)言本身,基本是做不了什么事的,必須要使用第三方擴(kuò)展庫(kù)。
這里羅列了Go語(yǔ)言的第三方庫(kù),通過(guò)這些第三方庫(kù)的介紹,我們也可以大概知道Go可以用來(lái)干什么。
如果你知道庫(kù)的名字的話,也可以在這個(gè)網(wǎng)站上搜索。
據(jù)我了解,Go的第三方庫(kù)大多都在GitHub上面。
#### 第六步:用Go做Web開發(fā)
Go是靜態(tài)語(yǔ)言,而且支持并發(fā)編程,所以,他有天然的性能優(yōu)勢(shì),大多公司主要使用Go也是開發(fā)后端服務(wù)(即API)。
終于到了實(shí)戰(zhàn)階段,如果我們真的要掌握一門語(yǔ)言,那么一定要用它來(lái)開發(fā)一個(gè)項(xiàng)目出來(lái)。這個(gè)過(guò)程大概需要一個(gè)月。
Beego是Go下在主流的Web開發(fā)框架,資料相對(duì)比較豐富,而且有完善的文檔。你可為此制定一個(gè)目標(biāo),比如用它來(lái)開發(fā)一個(gè)Blog,為此,你需要詳細(xì)閱讀Beego文檔,以及學(xué)習(xí)相關(guān)的Web開發(fā)技術(shù)。
等你完成這個(gè)項(xiàng)目的時(shí)候,我想你已經(jīng)會(huì)使用Go語(yǔ)言了。
從上一節(jié)的內(nèi)容可知,Do() 和 Receive() 等方法的返回值,除了 error 外,是一個(gè) interface{} 類型的返回值,因此當(dāng)我們的復(fù)雜操作返回的不是基本數(shù)據(jù)類型時(shí),就需要我們自己解析返回值,例如,當(dāng)我們利用 HMGET 方法獲取一批返回值時(shí),就需要對(duì)返回結(jié)果進(jìn)行解析,具體如下:
由于返回值是多條數(shù)據(jù),因此需要先將 reply 轉(zhuǎn)成 []interface 類型,然后在遍歷結(jié)果時(shí)在分別轉(zhuǎn)成 []uint8 (byte數(shù)組), 最后再轉(zhuǎn)成 string 類型。
隨著我們操作復(fù)雜度,數(shù)據(jù)解析的工作量也會(huì)非常大,(lua 腳本的使用,會(huì)使結(jié)果的解析更為復(fù)雜,因?yàn)榭赡艽嬖诙喾N類型的結(jié)果一起返回的情況,lua 腳本相關(guān)的內(nèi)容會(huì)在下一節(jié)介紹)。
redigo 包中的返回值助手函數(shù)的存在,就是為了幫助我們完成這些枯燥繁瑣的數(shù)據(jù)解析過(guò)程。
返回值助手函數(shù)相關(guān)源碼路徑為 github.com/gomodule/redigo/redis/reply.go 提供的主要方法如下:
上述返回值助手函數(shù)的具體使用,應(yīng)該依據(jù)具體的命令進(jìn)行選擇。如果大家還記得上一節(jié)介紹的 Redis 基本數(shù)據(jù)類型,可能會(huì)有些疑問(wèn),對(duì)于 redis 來(lái)說(shuō),其數(shù)據(jù)據(jù)存儲(chǔ)本質(zhì)都是 []bytes, 為什么可以解析出 Int、int64、float等類型的數(shù)據(jù)呢?
我們以 Float64() 為例進(jìn)行說(shuō)明,具體源碼如下:
其實(shí),返回值助手函數(shù)是將 []byte 類型的原始數(shù)據(jù),利用 strconv.ParseFloat(string(reply), 64) 轉(zhuǎn)換成了 float64類型,因此在我們使用過(guò)程中返回值助手函數(shù)的選擇,應(yīng)該基于業(yè)務(wù)和實(shí)際存儲(chǔ)的數(shù)據(jù)格式為依據(jù)。我們以第一小節(jié)的示例為例,看返回值助手函數(shù)如何降低我們的工作量,具體如下:
除了使用返回值助手函數(shù)對(duì)上述固定結(jié)構(gòu)的結(jié)果進(jìn)行解析外,redigo 包還提供了一個(gè) Scan()函數(shù)用于解析自定義的復(fù)雜數(shù)據(jù)結(jié)構(gòu),我們依然以上一個(gè)示例進(jìn)行說(shuō)明,具體示例如下:
如果返回結(jié)果為結(jié)構(gòu)化切片,也可以使用 canSlice() 方法,從而簡(jiǎn)化 loop 處理的部分,具體示例如下:
通過(guò)上述的示例,我們介紹了 scan 函數(shù)的基本用法,但是細(xì)心的同學(xué)可能會(huì)發(fā)現(xiàn)嗎,為什么數(shù)據(jù)寫入時(shí),value 的類型為 []int64 但是讀取時(shí)只能按照 string 類型讀取呢。這是因?yàn)?Redis 底層存儲(chǔ)的數(shù)據(jù)本質(zhì)都是 string 類型,。 無(wú)論是 HMSET 還是 MSET 最終都只能按照 string 類型讀取,因?yàn)槠浔举|(zhì)都是 hash 結(jié)構(gòu),不同之處僅在于 HMSET 是嵌套的 hash類型。 因此,[]int64 數(shù)據(jù)在寫入階段,就已經(jīng)被自動(dòng)處理為 []byte,寫入 redis 之后,len 和 類型 屬性會(huì)丟失。
如果強(qiáng)行按照 []int64解析將出錯(cuò):
如果 value 必須以結(jié)構(gòu)化的數(shù)據(jù)存儲(chǔ),那么可以提前對(duì)要寫入的數(shù)據(jù)進(jìn)行編碼,例如 json、protobuf 等,取出后再進(jìn)行解碼獲得原始數(shù)據(jù)。
轉(zhuǎn)載請(qǐng)參見(jiàn)文章末尾處的要求?!靖兄x張佳偉(@ghosert)的熱心翻譯。如果其他朋友也有不錯(cuò)的原創(chuàng)或譯文,可以嘗試推薦給伯樂(lè)在線?!窟@是一篇(長(zhǎng))博文, 介紹了我們?cè)?Repustate 遷移大量 Python/Cython 代碼到 Go 語(yǔ)言的經(jīng)驗(yàn)。如果你想了解整個(gè)故事,背景和所有的事情,請(qǐng)繼續(xù)往下讀。如果你只是想了解 Python 開發(fā)者在一頭扎進(jìn) Go 語(yǔ)言前需要了解什么,請(qǐng)點(diǎn)擊一下鏈接:從Python遷移到Go的建議(Tips Tricks) 背景在Repustate,我們完成過(guò)的最棒的技術(shù)成就之一是實(shí)現(xiàn)了阿拉伯語(yǔ)的情感分析。阿拉伯語(yǔ)是一塊難啃的硬骨頭,因?yàn)樗脑~形變化相當(dāng)復(fù)雜。比起譬如英語(yǔ),阿拉伯語(yǔ)的分詞(將一個(gè)句子切分呈幾個(gè)獨(dú)立的單詞)也更困難,因?yàn)榘⒗Z(yǔ)的單詞本身還可能會(huì)包含空白字符(例如:“阿列夫”在一個(gè)單詞里的位置)。這也談不上是泄密,Repustate 使用支持向量機(jī)(SVM)來(lái)獲取一個(gè)句子背后最有可能的含義,并在其中加上情感元素。 總體上來(lái)說(shuō),我們使用了 22 種模型(22 個(gè) SVM) 并且在一篇文檔中,每一個(gè)單詞我們都會(huì)加以分析。因此如果你有一篇 500 字的文檔,那么基于 SVM,會(huì)進(jìn)行十萬(wàn)次的比較。 PythonRepustate 幾乎完全就是一個(gè) Python 商店。我們使用 Django 來(lái)實(shí)現(xiàn) API 和網(wǎng)站。因此(目前)為了保持代碼一致,同時(shí)使用 Python 來(lái)實(shí)現(xiàn)阿拉伯語(yǔ)情感引擎是合情合理的。只是做原型和實(shí)現(xiàn)的話,Python 是很好的選擇。它的表達(dá)能力很強(qiáng)悍,第三方類庫(kù)等等也很好。如果你就是為了Web服務(wù),Python 很完美。但是當(dāng)你進(jìn)行低級(jí)別的計(jì)算,大量依賴于哈希表(Python 里的字典類型)做比較的時(shí)候,一切都變慢了。我們每秒能處理大約兩到三個(gè)阿拉伯文檔,但是這太慢了。比較下來(lái),我們的英語(yǔ)情感引擎每秒能處理大約五百份文檔。 瓶頸因此我們開啟了 Python 分析器,開始調(diào)查是什么地方用了那么長(zhǎng)時(shí)間。還記得我前面說(shuō)過(guò)我們有 22 個(gè) SVM 并且每個(gè)單詞都需要經(jīng)過(guò)處理嗎?好吧,這些都是線性處理的,非并行處理。所以我們的第一反應(yīng)是把線性處理改成 map/reduce 那樣的操作。簡(jiǎn)單來(lái)說(shuō):Python 不太適合用作 map/reduce。當(dāng)你需要并發(fā)的時(shí)候,Python 算上好用。在 2013 Python 大會(huì)上(譯者:PyCon 2013),Guido 談到了 Tulip,他的這個(gè)新項(xiàng)目正在彌補(bǔ) Python 這方面的不足,不過(guò)得過(guò)段一段時(shí)間才能推出,但是如果已經(jīng)有了更好用的東西,我們?yōu)槭裁催€要等呢? 選Go 語(yǔ)言,還是回家算了?我在Mozilla的朋友告訴我,Mozilla 內(nèi)部正在將他們大量的基礎(chǔ)日志架構(gòu)切換到 Go 語(yǔ)言上,部分原因是因?yàn)閺?qiáng)大的 [goroutines]。Go 語(yǔ)言是 Google 的人設(shè)計(jì)的,并且在設(shè)計(jì)之初就把支持并發(fā)作為第一要?jiǎng)?wù),而不是像 Python 的各種解決方案那樣是事后才加上去的。因此我們開始著手把 Python 換成 Go 語(yǔ)言。雖然Go 代碼還不算正式上線的產(chǎn)品,但是結(jié)果非常令人鼓舞。我們現(xiàn)在能做到每秒處理一千份文檔,使用更少的內(nèi)存,還不用調(diào)試你在 Python 里遇到:丑陋的多進(jìn)程/gevent/“為什么 Control-C 殺不了進(jìn)程”這些問(wèn)題。 為什么我們喜歡 Go 語(yǔ)言任何人,對(duì)編程語(yǔ)言是如何工作(解釋型 vs 編譯型, 動(dòng)態(tài)語(yǔ)言 vs 靜態(tài)語(yǔ)言)有一點(diǎn)理解的話,會(huì)說(shuō),“切,當(dāng)然 Go 語(yǔ)言會(huì)更快”。是的,我們也可以用 Java 把所有的東西重寫一遍,也能看到類似更快的改善,但那不是 Go 語(yǔ)言勝出的原因。你用 Go 寫的代碼好像就是對(duì)的。我搞不清楚到底是怎么回事,但是一旦代碼被編譯了(編譯速度很快),你就會(huì)覺(jué)得這代碼能工作(不只是跑起來(lái)不會(huì)錯(cuò),而且甚至邏輯上也是對(duì)的)。我知道,這聽(tīng)上去不太靠譜,但是確實(shí)如此。這和 Python 在冗余(或非冗余)方面非常類似,它把函數(shù)作為第一目標(biāo),因此函數(shù)編程會(huì)很容易想明白。而且當(dāng)然,go 線程和通道讓你的生活更容易,你可以得到靜態(tài)類型帶來(lái)的性能大提升,還能更精細(xì)的控制內(nèi)存分配,而你卻不必為此在語(yǔ)言表達(dá)力上付出太多的代價(jià)。 希望能早點(diǎn)知道的事情(Tips Tricks)除去所有這些贊美之詞以后,有時(shí)你真的需要在處理 Go 代碼的時(shí)候,相對(duì)于 Python,改變一下思維方式。因此這是我在遷移代碼時(shí)記錄的筆記清單 —— 只是在我把 Python 代碼轉(zhuǎn)換到 Go 時(shí)從我腦子里隨機(jī)冒出來(lái)的點(diǎn)子:沒(méi)有內(nèi)建的集合類型(必須使用map,并檢查是否存在)因?yàn)闆](méi)有集合,必須自己寫交集,并集之類的方法沒(méi)有tuples 類型,必須寫你自己的結(jié)構(gòu),或者使用 slices (即數(shù)組)沒(méi)有類似 \__getattr__() 的方法,你必須總是檢查存在性,而不是設(shè)置默認(rèn)值,例如,在 Python 里,你可以這樣寫 value = dict.get(“a_key”, “default_value”)必須總是檢查錯(cuò)誤(或者顯式的忽略錯(cuò)誤)不能有變量/包沒(méi)被使用,因此簡(jiǎn)單的測(cè)試也需要有時(shí)注掉一些代碼在[] byte 和 string 之間轉(zhuǎn)換。 regexp 使用 [] byte (不可變)。這是對(duì)的,但是老把一些變量轉(zhuǎn)換來(lái)轉(zhuǎn)換去很煩人Python 更寬松。你可以使用超出范圍的索引在字符串里取一個(gè)片段,而且不會(huì)出錯(cuò)。你還可以用負(fù)數(shù)取出片段,但是 Go 不行你不能混合數(shù)據(jù)結(jié)構(gòu)類型。也許這樣也不太干凈,但是有時(shí)在 Python 里,我會(huì)使用值是混合了字符串和列表的字典。但是 Go 不行,你不得不清理干凈你的數(shù)據(jù)結(jié)構(gòu)或者使用自定義的結(jié)構(gòu)不能解包一個(gè) tuple 或者 list 到幾個(gè)不同的變量(例如:x, y, z = [1, 2, 3])駝峰式命名風(fēng)格(如果你沒(méi)有首字大寫方法名/結(jié)構(gòu)名,他們不會(huì)被暴露給其它的包)。我更喜歡 Python 的小寫字母加下劃線命名風(fēng)格。必須顯式檢查是否有錯(cuò)誤 != nil, 不像在 Python 里,許多類型可以像 bool 那樣檢查 (0, “”, None 都可以被解釋成 “非” 集合)文檔在一些模塊上太散亂了,例如(crypto/md5),但是 IRC 上的 go-nuts 很好用,提供了巨大的幫助。從數(shù)字到字符串的轉(zhuǎn)換(int64 - string) 和 []byte - string (只要使用 string([]byte))不太一樣。需要使用 strconv。閱讀Go 代碼比起 Python 那樣寫起來(lái)如偽代碼的語(yǔ)言更像一門編程語(yǔ)言, Go 有更多的非字母數(shù)字字符,并且使用 || 和 , 而不是 “or”和“and”寫一個(gè)文件的話,有 File.Write([]byte) 和 File.WriteString(string), 這點(diǎn)和 Python 開發(fā)者的 Python 之道:“解決問(wèn)題就一種方法 ”相違背。修改字符串很困難,必須經(jīng)常重排 fmt.Sprintf沒(méi)有構(gòu)造函數(shù),因此慣用法是創(chuàng)建 NewType() 方法來(lái)返回你要的結(jié)構(gòu)Else (或者 else if)必須正確格式化,else 得和 if 配對(duì)的大括號(hào)在同一行。奇怪。賦值運(yùn)算符取決于在函數(shù)內(nèi)還是函數(shù)外,例如,= 和 :=如果我只想要“鍵”或者只想要 “值”,譬如: dict.keys() 或者 dict.values(),或者一個(gè) tuples 的列表,例如:dict.items(),在 Go 語(yǔ)言里沒(méi)有等價(jià)的東西,你只能自己枚舉 map 來(lái)構(gòu)造你的列表類型我有時(shí)使用一種習(xí)慣用法:構(gòu)造一個(gè)值是函數(shù)的字典類型,我想通過(guò)給定的鍵值調(diào)用這些函數(shù),你在 Go 里可以做到,但是所有的函數(shù)必須接受,返回相同的東西,例如:相同的方法簽名如果你使用 JSON 并且 你的 JSON 是一個(gè)復(fù)合類型,恭喜你。 你必須構(gòu)造自定義的結(jié)構(gòu)匹配 JSON 塊里的格式,然后把原始 JSON 解析到你自定義結(jié)構(gòu)的實(shí)例中去。比起 Python 世界里 object = json.loads(json_blob) 要做更多的工作 是不是值得?值得,一百萬(wàn)倍的值得。速度的提升太多了,以致很難舍棄。同時(shí),我認(rèn)為, Go 是目前趨勢(shì)所在,因此在招新員工的時(shí)候,我認(rèn)為把 Go 當(dāng)作 Repustate 技術(shù)積累的重要一環(huán)會(huì)很有幫助。]
換行符 \n 在 Windows 記事本不會(huì)顯示,用 Notepad2、Notepad++、UltraEdit 等打開就能看到,或者用 \r\n